|  | //===--- SemaAttr.cpp - Semantic Analysis for Attributes ------------------===// | 
|  | // | 
|  | //                     The LLVM Compiler Infrastructure | 
|  | // | 
|  | // This file is distributed under the University of Illinois Open Source | 
|  | // License. See LICENSE.TXT for details. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  | // | 
|  | // This file implements semantic analysis for non-trivial attributes and | 
|  | // pragmas. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "clang/Sema/SemaInternal.h" | 
|  | #include "clang/Sema/Lookup.h" | 
|  | #include "clang/AST/Attr.h" | 
|  | #include "clang/AST/Expr.h" | 
|  | #include "clang/Basic/TargetInfo.h" | 
|  | #include "clang/Lex/Preprocessor.h" | 
|  | using namespace clang; | 
|  |  | 
|  | //===----------------------------------------------------------------------===// | 
|  | // Pragma 'pack' and 'options align' | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | namespace { | 
|  | struct PackStackEntry { | 
|  | // We just use a sentinel to represent when the stack is set to mac68k | 
|  | // alignment. | 
|  | static const unsigned kMac68kAlignmentSentinel = ~0U; | 
|  |  | 
|  | unsigned Alignment; | 
|  | IdentifierInfo *Name; | 
|  | }; | 
|  |  | 
|  | /// PragmaPackStack - Simple class to wrap the stack used by #pragma | 
|  | /// pack. | 
|  | class PragmaPackStack { | 
|  | typedef std::vector<PackStackEntry> stack_ty; | 
|  |  | 
|  | /// Alignment - The current user specified alignment. | 
|  | unsigned Alignment; | 
|  |  | 
|  | /// Stack - Entries in the #pragma pack stack, consisting of saved | 
|  | /// alignments and optional names. | 
|  | stack_ty Stack; | 
|  |  | 
|  | public: | 
|  | PragmaPackStack() : Alignment(0) {} | 
|  |  | 
|  | void setAlignment(unsigned A) { Alignment = A; } | 
|  | unsigned getAlignment() { return Alignment; } | 
|  |  | 
|  | /// push - Push the current alignment onto the stack, optionally | 
|  | /// using the given \arg Name for the record, if non-zero. | 
|  | void push(IdentifierInfo *Name) { | 
|  | PackStackEntry PSE = { Alignment, Name }; | 
|  | Stack.push_back(PSE); | 
|  | } | 
|  |  | 
|  | /// pop - Pop a record from the stack and restore the current | 
|  | /// alignment to the previous value. If \arg Name is non-zero then | 
|  | /// the first such named record is popped, otherwise the top record | 
|  | /// is popped. Returns true if the pop succeeded. | 
|  | bool pop(IdentifierInfo *Name, bool IsReset); | 
|  | }; | 
|  | }  // end anonymous namespace. | 
|  |  | 
|  | bool PragmaPackStack::pop(IdentifierInfo *Name, bool IsReset) { | 
|  | // If name is empty just pop top. | 
|  | if (!Name) { | 
|  | // An empty stack is a special case... | 
|  | if (Stack.empty()) { | 
|  | // If this isn't a reset, it is always an error. | 
|  | if (!IsReset) | 
|  | return false; | 
|  |  | 
|  | // Otherwise, it is an error only if some alignment has been set. | 
|  | if (!Alignment) | 
|  | return false; | 
|  |  | 
|  | // Otherwise, reset to the default alignment. | 
|  | Alignment = 0; | 
|  | } else { | 
|  | Alignment = Stack.back().Alignment; | 
|  | Stack.pop_back(); | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // Otherwise, find the named record. | 
|  | for (unsigned i = Stack.size(); i != 0; ) { | 
|  | --i; | 
|  | if (Stack[i].Name == Name) { | 
|  | // Found it, pop up to and including this record. | 
|  | Alignment = Stack[i].Alignment; | 
|  | Stack.erase(Stack.begin() + i, Stack.end()); | 
|  | return true; | 
|  | } | 
|  | } | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  |  | 
|  | /// FreePackedContext - Deallocate and null out PackContext. | 
|  | void Sema::FreePackedContext() { | 
|  | delete static_cast<PragmaPackStack*>(PackContext); | 
|  | PackContext = 0; | 
|  | } | 
|  |  | 
|  | void Sema::AddAlignmentAttributesForRecord(RecordDecl *RD) { | 
|  | // If there is no pack context, we don't need any attributes. | 
|  | if (!PackContext) | 
|  | return; | 
|  |  | 
|  | PragmaPackStack *Stack = static_cast<PragmaPackStack*>(PackContext); | 
|  |  | 
|  | // Otherwise, check to see if we need a max field alignment attribute. | 
|  | if (unsigned Alignment = Stack->getAlignment()) { | 
|  | if (Alignment == PackStackEntry::kMac68kAlignmentSentinel) | 
|  | RD->addAttr(::new (Context) AlignMac68kAttr(SourceLocation(), Context)); | 
|  | else | 
|  | RD->addAttr(::new (Context) MaxFieldAlignmentAttr(SourceLocation(), | 
|  | Context, | 
|  | Alignment * 8)); | 
|  | } | 
|  | } | 
|  |  | 
|  | void Sema::ActOnPragmaOptionsAlign(PragmaOptionsAlignKind Kind, | 
|  | SourceLocation PragmaLoc, | 
|  | SourceLocation KindLoc) { | 
|  | if (PackContext == 0) | 
|  | PackContext = new PragmaPackStack(); | 
|  |  | 
|  | PragmaPackStack *Context = static_cast<PragmaPackStack*>(PackContext); | 
|  |  | 
|  | // Reset just pops the top of the stack, or resets the current alignment to | 
|  | // default. | 
|  | if (Kind == Sema::POAK_Reset) { | 
|  | if (!Context->pop(0, /*IsReset=*/true)) { | 
|  | Diag(PragmaLoc, diag::warn_pragma_options_align_reset_failed) | 
|  | << "stack empty"; | 
|  | } | 
|  | return; | 
|  | } | 
|  |  | 
|  | switch (Kind) { | 
|  | // For all targets we support native and natural are the same. | 
|  | // | 
|  | // FIXME: This is not true on Darwin/PPC. | 
|  | case POAK_Native: | 
|  | case POAK_Power: | 
|  | case POAK_Natural: | 
|  | Context->push(0); | 
|  | Context->setAlignment(0); | 
|  | break; | 
|  |  | 
|  | // Note that '#pragma options align=packed' is not equivalent to attribute | 
|  | // packed, it has a different precedence relative to attribute aligned. | 
|  | case POAK_Packed: | 
|  | Context->push(0); | 
|  | Context->setAlignment(1); | 
|  | break; | 
|  |  | 
|  | case POAK_Mac68k: | 
|  | // Check if the target supports this. | 
|  | if (!PP.getTargetInfo().hasAlignMac68kSupport()) { | 
|  | Diag(PragmaLoc, diag::err_pragma_options_align_mac68k_target_unsupported); | 
|  | return; | 
|  | } | 
|  | Context->push(0); | 
|  | Context->setAlignment(PackStackEntry::kMac68kAlignmentSentinel); | 
|  | break; | 
|  |  | 
|  | default: | 
|  | Diag(PragmaLoc, diag::warn_pragma_options_align_unsupported_option) | 
|  | << KindLoc; | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | void Sema::ActOnPragmaPack(PragmaPackKind Kind, IdentifierInfo *Name, | 
|  | ExprTy *alignment, SourceLocation PragmaLoc, | 
|  | SourceLocation LParenLoc, SourceLocation RParenLoc) { | 
|  | Expr *Alignment = static_cast<Expr *>(alignment); | 
|  |  | 
|  | // If specified then alignment must be a "small" power of two. | 
|  | unsigned AlignmentVal = 0; | 
|  | if (Alignment) { | 
|  | llvm::APSInt Val; | 
|  |  | 
|  | // pack(0) is like pack(), which just works out since that is what | 
|  | // we use 0 for in PackAttr. | 
|  | if (Alignment->isTypeDependent() || | 
|  | Alignment->isValueDependent() || | 
|  | !Alignment->isIntegerConstantExpr(Val, Context) || | 
|  | !(Val == 0 || Val.isPowerOf2()) || | 
|  | Val.getZExtValue() > 16) { | 
|  | Diag(PragmaLoc, diag::warn_pragma_pack_invalid_alignment); | 
|  | return; // Ignore | 
|  | } | 
|  |  | 
|  | AlignmentVal = (unsigned) Val.getZExtValue(); | 
|  | } | 
|  |  | 
|  | if (PackContext == 0) | 
|  | PackContext = new PragmaPackStack(); | 
|  |  | 
|  | PragmaPackStack *Context = static_cast<PragmaPackStack*>(PackContext); | 
|  |  | 
|  | switch (Kind) { | 
|  | case Sema::PPK_Default: // pack([n]) | 
|  | Context->setAlignment(AlignmentVal); | 
|  | break; | 
|  |  | 
|  | case Sema::PPK_Show: // pack(show) | 
|  | // Show the current alignment, making sure to show the right value | 
|  | // for the default. | 
|  | AlignmentVal = Context->getAlignment(); | 
|  | // FIXME: This should come from the target. | 
|  | if (AlignmentVal == 0) | 
|  | AlignmentVal = 8; | 
|  | if (AlignmentVal == PackStackEntry::kMac68kAlignmentSentinel) | 
|  | Diag(PragmaLoc, diag::warn_pragma_pack_show) << "mac68k"; | 
|  | else | 
|  | Diag(PragmaLoc, diag::warn_pragma_pack_show) << AlignmentVal; | 
|  | break; | 
|  |  | 
|  | case Sema::PPK_Push: // pack(push [, id] [, [n]) | 
|  | Context->push(Name); | 
|  | // Set the new alignment if specified. | 
|  | if (Alignment) | 
|  | Context->setAlignment(AlignmentVal); | 
|  | break; | 
|  |  | 
|  | case Sema::PPK_Pop: // pack(pop [, id] [,  n]) | 
|  | // MSDN, C/C++ Preprocessor Reference > Pragma Directives > pack: | 
|  | // "#pragma pack(pop, identifier, n) is undefined" | 
|  | if (Alignment && Name) | 
|  | Diag(PragmaLoc, diag::warn_pragma_pack_pop_identifer_and_alignment); | 
|  |  | 
|  | // Do the pop. | 
|  | if (!Context->pop(Name, /*IsReset=*/false)) { | 
|  | // If a name was specified then failure indicates the name | 
|  | // wasn't found. Otherwise failure indicates the stack was | 
|  | // empty. | 
|  | Diag(PragmaLoc, diag::warn_pragma_pack_pop_failed) | 
|  | << (Name ? "no record matching name" : "stack empty"); | 
|  |  | 
|  | // FIXME: Warn about popping named records as MSVC does. | 
|  | } else { | 
|  | // Pop succeeded, set the new alignment if specified. | 
|  | if (Alignment) | 
|  | Context->setAlignment(AlignmentVal); | 
|  | } | 
|  | break; | 
|  |  | 
|  | default: | 
|  | assert(0 && "Invalid #pragma pack kind."); | 
|  | } | 
|  | } | 
|  |  | 
|  | void Sema::ActOnPragmaUnused(const Token &IdTok, Scope *curScope, | 
|  | SourceLocation PragmaLoc) { | 
|  |  | 
|  | IdentifierInfo *Name = IdTok.getIdentifierInfo(); | 
|  | LookupResult Lookup(*this, Name, IdTok.getLocation(), LookupOrdinaryName); | 
|  | LookupParsedName(Lookup, curScope, NULL, true); | 
|  |  | 
|  | if (Lookup.empty()) { | 
|  | Diag(PragmaLoc, diag::warn_pragma_unused_undeclared_var) | 
|  | << Name << SourceRange(IdTok.getLocation()); | 
|  | return; | 
|  | } | 
|  |  | 
|  | VarDecl *VD = Lookup.getAsSingle<VarDecl>(); | 
|  | if (!VD) { | 
|  | Diag(PragmaLoc, diag::warn_pragma_unused_expected_var_arg) | 
|  | << Name << SourceRange(IdTok.getLocation()); | 
|  | return; | 
|  | } | 
|  |  | 
|  | // Warn if this was used before being marked unused. | 
|  | if (VD->isUsed()) | 
|  | Diag(PragmaLoc, diag::warn_used_but_marked_unused) << Name; | 
|  |  | 
|  | VD->addAttr(::new (Context) UnusedAttr(IdTok.getLocation(), Context)); | 
|  | } | 
|  |  | 
|  | typedef std::vector<std::pair<unsigned, SourceLocation> > VisStack; | 
|  | enum { NoVisibility = (unsigned) -1 }; | 
|  |  | 
|  | void Sema::AddPushedVisibilityAttribute(Decl *D) { | 
|  | if (!VisContext) | 
|  | return; | 
|  |  | 
|  | if (D->hasAttr<VisibilityAttr>()) | 
|  | return; | 
|  |  | 
|  | VisStack *Stack = static_cast<VisStack*>(VisContext); | 
|  | unsigned rawType = Stack->back().first; | 
|  | if (rawType == NoVisibility) return; | 
|  |  | 
|  | VisibilityAttr::VisibilityType type | 
|  | = (VisibilityAttr::VisibilityType) rawType; | 
|  | SourceLocation loc = Stack->back().second; | 
|  |  | 
|  | D->addAttr(::new (Context) VisibilityAttr(loc, Context, type)); | 
|  | } | 
|  |  | 
|  | /// FreeVisContext - Deallocate and null out VisContext. | 
|  | void Sema::FreeVisContext() { | 
|  | delete static_cast<VisStack*>(VisContext); | 
|  | VisContext = 0; | 
|  | } | 
|  |  | 
|  | static void PushPragmaVisibility(Sema &S, unsigned type, SourceLocation loc) { | 
|  | // Put visibility on stack. | 
|  | if (!S.VisContext) | 
|  | S.VisContext = new VisStack; | 
|  |  | 
|  | VisStack *Stack = static_cast<VisStack*>(S.VisContext); | 
|  | Stack->push_back(std::make_pair(type, loc)); | 
|  | } | 
|  |  | 
|  | void Sema::ActOnPragmaVisibility(bool IsPush, const IdentifierInfo* VisType, | 
|  | SourceLocation PragmaLoc) { | 
|  | if (IsPush) { | 
|  | // Compute visibility to use. | 
|  | VisibilityAttr::VisibilityType type; | 
|  | if (VisType->isStr("default")) | 
|  | type = VisibilityAttr::Default; | 
|  | else if (VisType->isStr("hidden")) | 
|  | type = VisibilityAttr::Hidden; | 
|  | else if (VisType->isStr("internal")) | 
|  | type = VisibilityAttr::Hidden; // FIXME | 
|  | else if (VisType->isStr("protected")) | 
|  | type = VisibilityAttr::Protected; | 
|  | else { | 
|  | Diag(PragmaLoc, diag::warn_attribute_unknown_visibility) << | 
|  | VisType->getName(); | 
|  | return; | 
|  | } | 
|  | PushPragmaVisibility(*this, type, PragmaLoc); | 
|  | } else { | 
|  | PopPragmaVisibility(); | 
|  | } | 
|  | } | 
|  |  | 
|  | void Sema::ActOnPragmaFPContract(tok::OnOffSwitch OOS) { | 
|  | switch (OOS) { | 
|  | case tok::OOS_ON: | 
|  | FPFeatures.fp_contract = 1; | 
|  | break; | 
|  | case tok::OOS_OFF: | 
|  | FPFeatures.fp_contract = 0; | 
|  | break; | 
|  | case tok::OOS_DEFAULT: | 
|  | FPFeatures.fp_contract = getLangOptions().DefaultFPContract; | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | void Sema::PushNamespaceVisibilityAttr(const VisibilityAttr *Attr) { | 
|  | // Visibility calculations will consider the namespace's visibility. | 
|  | // Here we just want to note that we're in a visibility context | 
|  | // which overrides any enclosing #pragma context, but doesn't itself | 
|  | // contribute visibility. | 
|  | PushPragmaVisibility(*this, NoVisibility, SourceLocation()); | 
|  | } | 
|  |  | 
|  | void Sema::PopPragmaVisibility() { | 
|  | // Pop visibility from stack, if there is one on the stack. | 
|  | if (VisContext) { | 
|  | VisStack *Stack = static_cast<VisStack*>(VisContext); | 
|  |  | 
|  | Stack->pop_back(); | 
|  | // To simplify the implementation, never keep around an empty stack. | 
|  | if (Stack->empty()) | 
|  | FreeVisContext(); | 
|  | } | 
|  | // FIXME: Add diag for pop without push. | 
|  | } |