Skip to content

Thread safety analysis: Fix a bug in handling temporary constructors #74020

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merged
merged 1 commit into from
Dec 8, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
26 changes: 13 additions & 13 deletions clang/lib/Analysis/ThreadSafety.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1010,6 +1010,8 @@ class ThreadSafetyAnalyzer {
ThreadSafetyHandler &Handler;
const FunctionDecl *CurrentFunction;
LocalVariableMap LocalVarMap;
// Maps constructed objects to `this` placeholder prior to initialization.
llvm::SmallDenseMap<const Expr *, til::LiteralPtr *> ConstructedObjects;
FactManager FactMan;
std::vector<CFGBlockInfo> BlockInfo;

Expand Down Expand Up @@ -1543,8 +1545,6 @@ class BuildLockset : public ConstStmtVisitor<BuildLockset> {
FactSet FSet;
// The fact set for the function on exit.
const FactSet &FunctionExitFSet;
/// Maps constructed objects to `this` placeholder prior to initialization.
llvm::SmallDenseMap<const Expr *, til::LiteralPtr *> ConstructedObjects;
LocalVariableMap::Context LVarCtx;
unsigned CtxIndex;

Expand Down Expand Up @@ -1808,7 +1808,7 @@ void BuildLockset::handleCall(const Expr *Exp, const NamedDecl *D,
std::pair<til::LiteralPtr *, StringRef> Placeholder =
Analyzer->SxBuilder.createThisPlaceholder(Exp);
[[maybe_unused]] auto inserted =
ConstructedObjects.insert({Exp, Placeholder.first});
Analyzer->ConstructedObjects.insert({Exp, Placeholder.first});
assert(inserted.second && "Are we visiting the same expression again?");
if (isa<CXXConstructExpr>(Exp))
Self = Placeholder.first;
Expand Down Expand Up @@ -2128,10 +2128,10 @@ void BuildLockset::VisitDeclStmt(const DeclStmt *S) {
E = EWC->getSubExpr()->IgnoreParens();
E = UnpackConstruction(E);

if (auto Object = ConstructedObjects.find(E);
Object != ConstructedObjects.end()) {
if (auto Object = Analyzer->ConstructedObjects.find(E);
Object != Analyzer->ConstructedObjects.end()) {
Object->second->setClangDecl(VD);
ConstructedObjects.erase(Object);
Analyzer->ConstructedObjects.erase(Object);
}
}
}
Expand All @@ -2140,11 +2140,11 @@ void BuildLockset::VisitDeclStmt(const DeclStmt *S) {
void BuildLockset::VisitMaterializeTemporaryExpr(
const MaterializeTemporaryExpr *Exp) {
if (const ValueDecl *ExtD = Exp->getExtendingDecl()) {
if (auto Object =
ConstructedObjects.find(UnpackConstruction(Exp->getSubExpr()));
Object != ConstructedObjects.end()) {
if (auto Object = Analyzer->ConstructedObjects.find(
UnpackConstruction(Exp->getSubExpr()));
Object != Analyzer->ConstructedObjects.end()) {
Object->second->setClangDecl(ExtD);
ConstructedObjects.erase(Object);
Analyzer->ConstructedObjects.erase(Object);
}
}
}
Expand Down Expand Up @@ -2487,15 +2487,15 @@ void ThreadSafetyAnalyzer::runAnalysis(AnalysisDeclContext &AC) {

// Clean up constructed object even if there are no attributes to
// keep the number of objects in limbo as small as possible.
if (auto Object = LocksetBuilder.ConstructedObjects.find(
if (auto Object = ConstructedObjects.find(
TD.getBindTemporaryExpr()->getSubExpr());
Object != LocksetBuilder.ConstructedObjects.end()) {
Object != ConstructedObjects.end()) {
const auto *DD = TD.getDestructorDecl(AC.getASTContext());
if (DD->hasAttrs())
// TODO: the location here isn't quite correct.
LocksetBuilder.handleCall(nullptr, DD, Object->second,
TD.getBindTemporaryExpr()->getEndLoc());
LocksetBuilder.ConstructedObjects.erase(Object);
ConstructedObjects.erase(Object);
}
break;
}
Expand Down
8 changes: 8 additions & 0 deletions clang/test/SemaCXX/warn-thread-safety-analysis.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1702,6 +1702,8 @@ struct TestScopedLockable {

bool getBool();

bool lock2Bool(MutexLock);

void foo1() {
MutexLock mulock(&mu1);
a = 5;
Expand All @@ -1718,6 +1720,12 @@ struct TestScopedLockable {
MutexLock{&mu1}, a = 5;
}

void temporary_cfg(int x) {
// test the case where a pair of temporary Ctor and Dtor is in different CFG blocks
lock2Bool(MutexLock{&mu1}) || x;
MutexLock{&mu1}; // no-warn
}

void lifetime_extension() {
const MutexLock &mulock = MutexLock(&mu1);
a = 5;
Expand Down