Change the error when a '+=' follows a declaration to suggest a fixit to '=' instead of just suggesting a ';'.

Old error:
plusequaldeclare1.cc:3:8: error: expected ';' at end of declaration
  int x += 6;
       ^
       ;

New error:
plusequaldeclare1.cc:3:9: error: invalid '+=' at end of declaration; did you
      mean '='?
  int x += 6;
        ^~
        =

llvm-svn: 148433
This commit is contained in:
Richard Trieu 2012-01-18 22:54:52 +00:00
parent 94298a906a
commit c64d3230d2
6 changed files with 40 additions and 19 deletions

View File

@ -156,6 +156,8 @@ def err_invalid_token_after_toplevel_declarator : Error<
"expected ';' after top level declarator">;
def err_invalid_equalequal_after_declarator : Error<
"invalid '==' at end of declaration; did you mean '='?">;
def err_invalid_plusequal_after_declarator : Error<
"invalid '+=' at end of declaration; did you mean '='?">;
def err_expected_statement : Error<"expected statement">;
def err_expected_lparen_after : Error<"expected '(' after '%0'">;
def err_expected_lparen_after_id : Error<"expected '(' after %0">;

View File

@ -288,10 +288,12 @@ private:
Tok.getKind() == tok::utf32_string_literal;
}
/// \brief Returns true if the current token is a '=' or '==' and
/// false otherwise. If it's '==', we assume that it's a typo and we emit
/// DiagID and a fixit hint to turn '==' -> '='.
bool isTokenEqualOrMistypedEqualEqual(unsigned DiagID);
/// \brief Returns true if the current token is FoundToken. This token
/// will be assumed a typo. A diagnostic will be emitted with DiagID with a
/// a fixit to replace the current token with ExpectedToken.
bool CreateTokenReplacement(tok::TokenKind ExpectedToken,
tok::TokenKind FoundToken,
unsigned DiagID);
/// ConsumeToken - Consume the current 'peek token' and lex the next one.
/// This does not work with all kinds of tokens: strings and specific other

View File

@ -1269,8 +1269,12 @@ Decl *Parser::ParseDeclarationAfterDeclaratorAndAttributes(Declarator &D,
D.getDeclSpec().getTypeSpecType() == DeclSpec::TST_auto;
// Parse declarator '=' initializer.
if (isTokenEqualOrMistypedEqualEqual(
diag::err_invalid_equalequal_after_declarator)) {
// If a '==' or '+=' is found, suggest a fixit to '='.
if (Tok.is(tok::equal) ||
CreateTokenReplacement(tok::equal, tok::equalequal,
diag::err_invalid_equalequal_after_declarator) ||
CreateTokenReplacement(tok::equal, tok::plusequal,
diag::err_invalid_plusequal_after_declarator)) {
ConsumeToken();
if (Tok.is(tok::kw_delete)) {
if (D.isFunctionDeclarator())

View File

@ -1259,8 +1259,12 @@ bool Parser::ParseCXXCondition(ExprResult &ExprOut,
ExprOut = ExprError();
// '=' assignment-expression
if (isTokenEqualOrMistypedEqualEqual(
diag::err_invalid_equalequal_after_declarator)) {
// If a '==' or '+=' is found, suggest a fixit to '='.
if (Tok.is(tok::equal) ||
CreateTokenReplacement(tok::equal, tok::equalequal,
diag::err_invalid_equalequal_after_declarator) ||
CreateTokenReplacement(tok::equal, tok::plusequal,
diag::err_invalid_plusequal_after_declarator)) {
ConsumeToken();
ExprResult AssignExpr(ParseAssignmentExpression());
if (!AssignExpr.isInvalid())

View File

@ -1401,20 +1401,23 @@ bool Parser::TryAnnotateCXXScopeToken(bool EnteringContext) {
return false;
}
bool Parser::isTokenEqualOrMistypedEqualEqual(unsigned DiagID) {
if (Tok.is(tok::equalequal)) {
// We have '==' in a context that we would expect a '='.
// The user probably made a typo, intending to type '='. Emit diagnostic,
// fixit hint to turn '==' -> '=' and continue as if the user typed '='.
bool Parser::CreateTokenReplacement(tok::TokenKind ExpectedToken,
tok::TokenKind FoundToken,
unsigned DiagID) {
if (Tok.isNot(FoundToken))
return false;
// We have FoundToken in a context that we would expect an ExpectedToken.
// The user probably made a typo, intending to type ExpectedToken.
// Emit diagnostic, fixit hint to turn ReplaceToken -> ExpectedToken
// and continue as if the user typed ExpectedToken.
Tok.setKind(ExpectedToken);
Diag(Tok, DiagID)
<< FixItHint::CreateReplacement(SourceRange(Tok.getLocation()),
getTokenSimpleSpelling(tok::equal));
getTokenSimpleSpelling(ExpectedToken));
return true;
}
return Tok.is(tok::equal);
}
SourceLocation Parser::handleUnexpectedCodeCompletionToken() {
assert(Tok.is(tok::code_completion));
PrevTokLocation = Tok.getLocation();

View File

@ -69,13 +69,19 @@ class C {
namespace rdar8488464 {
int x == 0; // expected-error {{invalid '==' at end of declaration; did you mean '='?}}
int y += 0; // expected-error {{invalid '+=' at end of declaration; did you mean '='?}}
void f() {
int x == 0; // expected-error {{invalid '==' at end of declaration; did you mean '='?}}
(void)x;
int y += 0; // expected-error {{invalid '+=' at end of declaration; did you mean '='?}}
(void)y;
if (int x == 0) { // expected-error {{invalid '==' at end of declaration; did you mean '='?}}
(void)x;
}
if (int y += 0) { // expected-error {{invalid '+=' at end of declaration; did you mean '='?}}
(void)y;
}
}
}