| //===--- 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 *Identifiers, unsigned NumIdentifiers, |
| Scope *curScope, |
| SourceLocation PragmaLoc, |
| SourceLocation LParenLoc, |
| SourceLocation RParenLoc) { |
| |
| for (unsigned i = 0; i < NumIdentifiers; ++i) { |
| const Token &Tok = Identifiers[i]; |
| IdentifierInfo *Name = Tok.getIdentifierInfo(); |
| LookupResult Lookup(*this, Name, Tok.getLocation(), LookupOrdinaryName); |
| LookupParsedName(Lookup, curScope, NULL, true); |
| |
| if (Lookup.empty()) { |
| Diag(PragmaLoc, diag::warn_pragma_unused_undeclared_var) |
| << Name << SourceRange(Tok.getLocation()); |
| continue; |
| } |
| |
| VarDecl *VD = Lookup.getAsSingle<VarDecl>(); |
| if (!VD || !VD->hasLocalStorage()) { |
| Diag(PragmaLoc, diag::warn_pragma_unused_expected_localvar) |
| << Name << SourceRange(Tok.getLocation()); |
| continue; |
| } |
| |
| // 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(Tok.getLocation(), Context)); |
| } |
| } |
| |
| typedef std::vector<std::pair<VisibilityAttr::VisibilityType, |
| SourceLocation> > VisStack; |
| |
| void Sema::AddPushedVisibilityAttribute(Decl *D) { |
| if (!VisContext) |
| return; |
| |
| if (D->hasAttr<VisibilityAttr>()) |
| return; |
| |
| VisStack *Stack = static_cast<VisStack*>(VisContext); |
| VisibilityAttr::VisibilityType type = Stack->back().first; |
| 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, VisibilityAttr::VisibilityType 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::PushVisibilityAttr(const VisibilityAttr *Attr) { |
| PushPragmaVisibility(*this, Attr->getVisibility(), Attr->getLocation()); |
| } |
| |
| 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. |
| } |