SemaStmtAttr.cpp revision 053214013990ad8ec096dafc64aa7c0ad2b05bc0
1//===--- SemaStmtAttr.cpp - Statement Attribute Handling ------------------===// 2// 3// The LLVM Compiler Infrastructure 4// 5// This file is distributed under the University of Illinois Open Source 6// License. See LICENSE.TXT for details. 7// 8//===----------------------------------------------------------------------===// 9// 10// This file implements stmt-related attribute processing. 11// 12//===----------------------------------------------------------------------===// 13 14#include "clang/Sema/SemaInternal.h" 15#include "TargetAttributesSema.h" 16#include "clang/AST/ASTContext.h" 17#include "clang/Basic/SourceManager.h" 18#include "clang/Lex/Lexer.h" 19#include "clang/Sema/DelayedDiagnostic.h" 20#include "clang/Sema/Lookup.h" 21#include "clang/Sema/ScopeInfo.h" 22#include "llvm/ADT/StringExtras.h" 23 24using namespace clang; 25using namespace sema; 26 27static Attr *handleFallThroughAttr(Sema &S, Stmt *St, const AttributeList &A, 28 SourceRange Range) { 29 if (!isa<NullStmt>(St)) { 30 S.Diag(A.getRange().getBegin(), diag::err_fallthrough_attr_wrong_target) 31 << St->getLocStart(); 32 if (isa<SwitchCase>(St)) { 33 SourceLocation L = Lexer::getLocForEndOfToken(Range.getEnd(), 0, 34 S.getSourceManager(), S.getLangOpts()); 35 S.Diag(L, diag::note_fallthrough_insert_semi_fixit) 36 << FixItHint::CreateInsertion(L, ";"); 37 } 38 return 0; 39 } 40 if (S.getCurFunction()->SwitchStack.empty()) { 41 S.Diag(A.getRange().getBegin(), diag::err_fallthrough_attr_outside_switch); 42 return 0; 43 } 44 return ::new (S.Context) FallThroughAttr(A.getRange(), S.Context); 45} 46 47 48static Attr *ProcessStmtAttribute(Sema &S, Stmt *St, const AttributeList &A, 49 SourceRange Range) { 50 switch (A.getKind()) { 51 case AttributeList::UnknownAttribute: 52 S.Diag(A.getLoc(), A.isDeclspecAttribute() ? 53 diag::warn_unhandled_ms_attribute_ignored : 54 diag::warn_unknown_attribute_ignored) << A.getName(); 55 return 0; 56 case AttributeList::AT_FallThrough: 57 return handleFallThroughAttr(S, St, A, Range); 58 default: 59 // if we're here, then we parsed a known attribute, but didn't recognize 60 // it as a statement attribute => it is declaration attribute 61 S.Diag(A.getRange().getBegin(), diag::err_attribute_invalid_on_stmt) 62 << A.getName() << St->getLocStart(); 63 return 0; 64 } 65} 66 67StmtResult Sema::ProcessStmtAttributes(Stmt *S, AttributeList *AttrList, 68 SourceRange Range) { 69 SmallVector<const Attr*, 8> Attrs; 70 for (const AttributeList* l = AttrList; l; l = l->getNext()) { 71 if (Attr *a = ProcessStmtAttribute(*this, S, *l, Range)) 72 Attrs.push_back(a); 73 } 74 75 if (Attrs.empty()) 76 return S; 77 78 return ActOnAttributedStmt(Range.getBegin(), Attrs, S); 79} 80