llvm-project/lldb/source/Symbol/CompilerDeclContext.cpp
Dave Lee 6c599b1e9b [lldb] Let 'v' command directly access ivars of _any_ self/this
The `v` (`frame variable`) command can directly access ivars/fields of `this` or `self`.
Such as `v field`, instead of `v this->field`. This change relaxes the criteria for
finding `this`/`self` variables.

There are cases where a `this`/`self` variable does exist, but up to now the `v` command
has not made use of it. The user would have to explicitly run `v this->field` or
`self->_ivar` to access ivars. This change allows such cases to also work (without
explicitly dereferencing `this`/`self`).

A very common example in Objective-C (and Swift) is weakly capturing `self`:

```
__weak Type *weakSelf = self;
void (^block)(void) = ^{
   Type *self = weakSelf; // Re-establish strong reference.
   // `v _ivar` should work just as well as `v self->_ivar`.
};
```

In this case, `self` exists but `v` would not have used it. With this change, the fact
that a variable named `self` exists is enough for it to be used.

Differential Revision: https://reviews.llvm.org/D145276
2023-03-08 11:19:43 -08:00

80 lines
2.7 KiB
C++

//===-- CompilerDeclContext.cpp -------------------------------------------===//
//
// 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 "lldb/Symbol/CompilerDeclContext.h"
#include "lldb/Symbol/CompilerDecl.h"
#include "lldb/Symbol/TypeSystem.h"
#include <vector>
using namespace lldb_private;
std::vector<CompilerDecl>
CompilerDeclContext::FindDeclByName(ConstString name,
const bool ignore_using_decls) {
if (IsValid())
return m_type_system->DeclContextFindDeclByName(m_opaque_decl_ctx, name,
ignore_using_decls);
return std::vector<CompilerDecl>();
}
ConstString CompilerDeclContext::GetName() const {
if (IsValid())
return m_type_system->DeclContextGetName(m_opaque_decl_ctx);
return ConstString();
}
ConstString CompilerDeclContext::GetScopeQualifiedName() const {
if (IsValid())
return m_type_system->DeclContextGetScopeQualifiedName(m_opaque_decl_ctx);
return ConstString();
}
bool CompilerDeclContext::IsClassMethod() {
if (IsValid())
return m_type_system->DeclContextIsClassMethod(m_opaque_decl_ctx);
return false;
}
lldb::LanguageType CompilerDeclContext::GetLanguage() {
if (IsValid())
return m_type_system->DeclContextGetLanguage(m_opaque_decl_ctx);
return {};
}
ConstString
CompilerDeclContext::GetInstanceVariableName(lldb::LanguageType language) {
if (IsValid())
return m_type_system->GetInstanceVariableName(language);
return {};
}
bool CompilerDeclContext::IsContainedInLookup(CompilerDeclContext other) const {
if (!IsValid())
return false;
// If the other context is just the current context, we don't need to go
// over the type system to know that the lookup is identical.
if (this == &other)
return true;
return m_type_system->DeclContextIsContainedInLookup(m_opaque_decl_ctx,
other.m_opaque_decl_ctx);
}
bool lldb_private::operator==(const lldb_private::CompilerDeclContext &lhs,
const lldb_private::CompilerDeclContext &rhs) {
return lhs.GetTypeSystem() == rhs.GetTypeSystem() &&
lhs.GetOpaqueDeclContext() == rhs.GetOpaqueDeclContext();
}
bool lldb_private::operator!=(const lldb_private::CompilerDeclContext &lhs,
const lldb_private::CompilerDeclContext &rhs) {
return lhs.GetTypeSystem() != rhs.GetTypeSystem() ||
lhs.GetOpaqueDeclContext() != rhs.GetOpaqueDeclContext();
}