| //===--- SemaStmtAttr.cpp - Statement Attribute Handling ------------------===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file implements stmt-related attribute processing. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "clang/Sema/SemaInternal.h" |
| #include "TargetAttributesSema.h" |
| #include "clang/AST/ASTContext.h" |
| #include "clang/Basic/SourceManager.h" |
| #include "clang/Lex/Lexer.h" |
| #include "clang/Sema/DelayedDiagnostic.h" |
| #include "clang/Sema/Lookup.h" |
| #include "clang/Sema/ScopeInfo.h" |
| #include "llvm/ADT/StringExtras.h" |
| |
| using namespace clang; |
| using namespace sema; |
| |
| static Attr *handleFallThroughAttr(Sema &S, Stmt *St, const AttributeList &A, |
| SourceRange Range) { |
| if (!isa<NullStmt>(St)) { |
| S.Diag(A.getRange().getBegin(), diag::err_fallthrough_attr_wrong_target) |
| << St->getLocStart(); |
| if (isa<SwitchCase>(St)) { |
| SourceLocation L = Lexer::getLocForEndOfToken(Range.getEnd(), 0, |
| S.getSourceManager(), S.getLangOpts()); |
| S.Diag(L, diag::note_fallthrough_insert_semi_fixit) |
| << FixItHint::CreateInsertion(L, ";"); |
| } |
| return 0; |
| } |
| if (S.getCurFunction()->SwitchStack.empty()) { |
| S.Diag(A.getRange().getBegin(), diag::err_fallthrough_attr_outside_switch); |
| return 0; |
| } |
| return ::new (S.Context) FallThroughAttr(A.getRange(), S.Context); |
| } |
| |
| |
| static Attr *ProcessStmtAttribute(Sema &S, Stmt *St, const AttributeList &A, |
| SourceRange Range) { |
| switch (A.getKind()) { |
| case AttributeList::AT_clang___fallthrough: |
| return handleFallThroughAttr(S, St, A, Range); |
| default: |
| // if we're here, then we parsed an attribute, but didn't recognize it as a |
| // statement attribute => it is declaration attribute |
| S.Diag(A.getRange().getBegin(), diag::warn_attribute_invalid_on_stmt) |
| << A.getName()->getName() << St->getLocStart(); |
| return 0; |
| } |
| } |
| |
| StmtResult Sema::ProcessStmtAttributes(Stmt *S, AttributeList *AttrList, |
| SourceRange Range) { |
| AttrVec Attrs; |
| for (const AttributeList* l = AttrList; l; l = l->getNext()) { |
| if (Attr *a = ProcessStmtAttribute(*this, S, *l, Range)) |
| Attrs.push_back(a); |
| } |
| |
| if (Attrs.empty()) |
| return S; |
| |
| return ActOnAttributedStmt(Range.getBegin(), Attrs, S); |
| } |