mirror of
https://github.com/capstone-engine/llvm-capstone.git
synced 2025-01-07 16:42:34 +00:00
Fix a think-o in handling ambiguous corrections for a TypoExpr.
Under certain circumstances, the identifier mentioned in the diagnostic won't match the intended correction even though the replacement expression and the note pointing to the decl are both correct. Basically, the TreeTransform assumes the TypoExpr's Consumer points to the correct TypoCorrection, but the handling of typos that appear to be ambiguous from the point of view of TransformTypoExpr would cause that assumption to be violated by altering the Consumer's correction stream. This fix allows the Consumer's correction stream to be reset to the right TypoCorrection after successfully resolving the percieved ambiguity. Included is a fix to suppress correcting the RHS of an assignment to the LHS of that assignment for non-C++ code, to prevent a regression in test/SemaObjC/provisional-ivar-lookup.m. This fixes PR22297. llvm-svn: 227251
This commit is contained in:
parent
1ac9356524
commit
7a503694fe
@ -101,7 +101,7 @@ public:
|
|||||||
DeclContext *MemberContext,
|
DeclContext *MemberContext,
|
||||||
bool EnteringContext)
|
bool EnteringContext)
|
||||||
: Typo(TypoName.getName().getAsIdentifierInfo()), CurrentTCIndex(0),
|
: Typo(TypoName.getName().getAsIdentifierInfo()), CurrentTCIndex(0),
|
||||||
SemaRef(SemaRef), S(S),
|
SavedTCIndex(0), SemaRef(SemaRef), S(S),
|
||||||
SS(SS ? llvm::make_unique<CXXScopeSpec>(*SS) : nullptr),
|
SS(SS ? llvm::make_unique<CXXScopeSpec>(*SS) : nullptr),
|
||||||
CorrectionValidator(std::move(CCC)), MemberContext(MemberContext),
|
CorrectionValidator(std::move(CCC)), MemberContext(MemberContext),
|
||||||
Result(SemaRef, TypoName, LookupKind),
|
Result(SemaRef, TypoName, LookupKind),
|
||||||
@ -187,6 +187,17 @@ public:
|
|||||||
CurrentTCIndex >= ValidatedCorrections.size();
|
CurrentTCIndex >= ValidatedCorrections.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// \brief Save the current position in the correction stream (overwriting any
|
||||||
|
/// previously saved position).
|
||||||
|
void saveCurrentPosition() {
|
||||||
|
SavedTCIndex = CurrentTCIndex;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// \brief Restore the saved position in the correction stream.
|
||||||
|
void restoreSavedPosition() {
|
||||||
|
CurrentTCIndex = SavedTCIndex;
|
||||||
|
}
|
||||||
|
|
||||||
ASTContext &getContext() const { return SemaRef.Context; }
|
ASTContext &getContext() const { return SemaRef.Context; }
|
||||||
const LookupResult &getLookupResult() const { return Result; }
|
const LookupResult &getLookupResult() const { return Result; }
|
||||||
|
|
||||||
@ -267,6 +278,7 @@ private:
|
|||||||
|
|
||||||
SmallVector<TypoCorrection, 4> ValidatedCorrections;
|
SmallVector<TypoCorrection, 4> ValidatedCorrections;
|
||||||
size_t CurrentTCIndex;
|
size_t CurrentTCIndex;
|
||||||
|
size_t SavedTCIndex;
|
||||||
|
|
||||||
Sema &SemaRef;
|
Sema &SemaRef;
|
||||||
Scope *S;
|
Scope *S;
|
||||||
|
@ -9459,6 +9459,18 @@ static void checkObjCPointerIntrospection(Sema &S, ExprResult &L, ExprResult &R,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static NamedDecl *getDeclFromExpr(Expr *E) {
|
||||||
|
if (!E)
|
||||||
|
return nullptr;
|
||||||
|
if (auto *DRE = dyn_cast<DeclRefExpr>(E))
|
||||||
|
return DRE->getDecl();
|
||||||
|
if (auto *ME = dyn_cast<MemberExpr>(E))
|
||||||
|
return ME->getMemberDecl();
|
||||||
|
if (auto *IRE = dyn_cast<ObjCIvarRefExpr>(E))
|
||||||
|
return IRE->getDecl();
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
/// CreateBuiltinBinOp - Creates a new built-in binary operation with
|
/// CreateBuiltinBinOp - Creates a new built-in binary operation with
|
||||||
/// operator @p Opc at location @c TokLoc. This routine only supports
|
/// operator @p Opc at location @c TokLoc. This routine only supports
|
||||||
/// built-in operations; ActOnBinOp handles overloaded operators.
|
/// built-in operations; ActOnBinOp handles overloaded operators.
|
||||||
@ -9496,7 +9508,13 @@ ExprResult Sema::CreateBuiltinBinOp(SourceLocation OpLoc,
|
|||||||
// doesn't handle dependent types properly, so make sure any TypoExprs have
|
// doesn't handle dependent types properly, so make sure any TypoExprs have
|
||||||
// been dealt with before checking the operands.
|
// been dealt with before checking the operands.
|
||||||
LHS = CorrectDelayedTyposInExpr(LHSExpr);
|
LHS = CorrectDelayedTyposInExpr(LHSExpr);
|
||||||
RHS = CorrectDelayedTyposInExpr(RHSExpr);
|
RHS = CorrectDelayedTyposInExpr(RHSExpr, [Opc, LHS](Expr *E) {
|
||||||
|
if (Opc != BO_Assign)
|
||||||
|
return ExprResult(E);
|
||||||
|
// Avoid correcting the RHS to the same Expr as the LHS.
|
||||||
|
Decl *D = getDeclFromExpr(E);
|
||||||
|
return (D && D == getDeclFromExpr(LHS.get())) ? ExprError() : E;
|
||||||
|
});
|
||||||
if (!LHS.isUsable() || !RHS.isUsable())
|
if (!LHS.isUsable() || !RHS.isUsable())
|
||||||
return ExprError();
|
return ExprError();
|
||||||
}
|
}
|
||||||
|
@ -6165,15 +6165,18 @@ public:
|
|||||||
while (!AmbiguousTypoExprs.empty()) {
|
while (!AmbiguousTypoExprs.empty()) {
|
||||||
auto TE = AmbiguousTypoExprs.back();
|
auto TE = AmbiguousTypoExprs.back();
|
||||||
auto Cached = TransformCache[TE];
|
auto Cached = TransformCache[TE];
|
||||||
AmbiguousTypoExprs.pop_back();
|
auto &State = SemaRef.getTypoExprState(TE);
|
||||||
|
State.Consumer->saveCurrentPosition();
|
||||||
TransformCache.erase(TE);
|
TransformCache.erase(TE);
|
||||||
if (!TryTransform(E).isInvalid()) {
|
if (!TryTransform(E).isInvalid()) {
|
||||||
SemaRef.getTypoExprState(TE).Consumer->resetCorrectionStream();
|
State.Consumer->resetCorrectionStream();
|
||||||
TransformCache.erase(TE);
|
TransformCache.erase(TE);
|
||||||
Res = ExprError();
|
Res = ExprError();
|
||||||
break;
|
break;
|
||||||
} else
|
}
|
||||||
TransformCache[TE] = Cached;
|
AmbiguousTypoExprs.remove(TE);
|
||||||
|
State.Consumer->restoreSavedPosition();
|
||||||
|
TransformCache[TE] = Cached;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure that all of the TypoExprs within the current Expr have been found.
|
// Ensure that all of the TypoExprs within the current Expr have been found.
|
||||||
|
@ -175,3 +175,13 @@ namespace PR22250 {
|
|||||||
// expected-error@+1 {{expected ';' after top level declarator}}
|
// expected-error@+1 {{expected ';' after top level declarator}}
|
||||||
int getenv_s(size_t *y, char(&z)) {}
|
int getenv_s(size_t *y, char(&z)) {}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
namespace PR22297 {
|
||||||
|
double pow(double x, double y);
|
||||||
|
struct TimeTicks {
|
||||||
|
static void Now(); // expected-note {{'Now' declared here}}
|
||||||
|
};
|
||||||
|
void f() {
|
||||||
|
TimeTicks::now(); // expected-error {{no member named 'now' in 'PR22297::TimeTicks'; did you mean 'Now'?}}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user