Files
clang-p2996/lldb/source/Plugins/ExpressionParser/Clang/ClangUtil.cpp
Adrian Prantl 6eaedbb52f Make CompilerType safe
When a process gets restarted TypeSystem objects associated with it
may get deleted, and any CompilerType objects holding on to a
reference to that type system are a use-after-free in waiting. Because
of the SBAPI, we don't have tight control over where CompilerTypes go
and when they are used. This is particularly a problem in the Swift
plugin, where the scratch TypeSystem can be restarted while the
process is still running. The Swift plugin has a lock to prevent
abuse, but where there's a lock there can be bugs.

This patch changes CompilerType to store a std::weak_ptr<TypeSystem>.
Most of the std::weak_ptr<TypeSystem>* uglyness is hidden by
introducing a wrapper class CompilerType::WrappedTypeSystem that has a
dyn_cast_or_null() method. The only sites that need to know about the
weak pointer implementation detail are the ones that deal with
creating TypeSystems.

rdar://101505232

Differential Revision: https://reviews.llvm.org/D136650
2022-11-16 15:51:26 -08:00

88 lines
2.4 KiB
C++

//===-- ClangUtil.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
//
// A collection of helper methods and data structures for manipulating clang
// types and decls.
//===----------------------------------------------------------------------===//
#include "Plugins/ExpressionParser/Clang/ClangUtil.h"
#include "Plugins/TypeSystem/Clang/TypeSystemClang.h"
using namespace clang;
using namespace lldb_private;
bool ClangUtil::IsClangType(const CompilerType &ct) {
// Invalid types are never Clang types.
if (!ct)
return false;
if (!ct.GetTypeSystem().dyn_cast_or_null<TypeSystemClang>())
return false;
if (!ct.GetOpaqueQualType())
return false;
return true;
}
clang::Decl *ClangUtil::GetDecl(const CompilerDecl &decl) {
assert(llvm::isa<TypeSystemClang>(decl.GetTypeSystem()));
return static_cast<clang::Decl *>(decl.GetOpaqueDecl());
}
QualType ClangUtil::GetQualType(const CompilerType &ct) {
// Make sure we have a clang type before making a clang::QualType
if (!IsClangType(ct))
return QualType();
return QualType::getFromOpaquePtr(ct.GetOpaqueQualType());
}
QualType ClangUtil::GetCanonicalQualType(const CompilerType &ct) {
if (!IsClangType(ct))
return QualType();
return GetQualType(ct).getCanonicalType();
}
CompilerType ClangUtil::RemoveFastQualifiers(const CompilerType &ct) {
if (!IsClangType(ct))
return ct;
QualType qual_type(GetQualType(ct));
qual_type.removeLocalFastQualifiers();
return CompilerType(ct.GetTypeSystem(), qual_type.getAsOpaquePtr());
}
clang::TagDecl *ClangUtil::GetAsTagDecl(const CompilerType &type) {
clang::QualType qual_type = ClangUtil::GetCanonicalQualType(type);
if (qual_type.isNull())
return nullptr;
return qual_type->getAsTagDecl();
}
std::string ClangUtil::DumpDecl(const clang::Decl *d) {
if (!d)
return "nullptr";
std::string result;
llvm::raw_string_ostream stream(result);
bool deserialize = false;
d->dump(stream, deserialize);
stream.flush();
return result;
}
std::string ClangUtil::ToString(const clang::Type *t) {
return clang::QualType(t, 0).getAsString();
}
std::string ClangUtil::ToString(const CompilerType &c) {
return ClangUtil::GetQualType(c).getAsString();
}