| //===--- AttributeList.cpp --------------------------------------*- C++ -*-===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file defines the AttributeList class implementation |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "clang/Parse/AttributeList.h" |
| #include "clang/Basic/IdentifierTable.h" |
| using namespace clang; |
| |
| AttributeList::AttributeList(IdentifierInfo *aName, SourceLocation aLoc, |
| IdentifierInfo *pName, SourceLocation pLoc, |
| ActionBase::ExprTy **ExprList, unsigned numArgs, |
| AttributeList *n) |
| : AttrName(aName), AttrLoc(aLoc), ParmName(pName), ParmLoc(pLoc), |
| NumArgs(numArgs), Next(n) { |
| |
| if (numArgs == 0) |
| Args = 0; |
| else { |
| Args = new ActionBase::ExprTy*[numArgs]; |
| memcpy(Args, ExprList, numArgs*sizeof(Args[0])); |
| } |
| } |
| |
| AttributeList::~AttributeList() { |
| if (Args) { |
| // FIXME: before we delete the vector, we need to make sure the Expr's |
| // have been deleted. Since ActionBase::ExprTy is "void", we are dependent |
| // on the actions module for actually freeing the memory. The specific |
| // hooks are ActOnDeclarator, ActOnTypeName, ActOnParamDeclaratorType, |
| // ParseField, ParseTag. Once these routines have freed the expression, |
| // they should zero out the Args slot (to indicate the memory has been |
| // freed). If any element of the vector is non-null, we should assert. |
| delete [] Args; |
| } |
| delete Next; |
| } |
| |
| AttributeList::Kind AttributeList::getKind(const IdentifierInfo *Name) { |
| const char *Str = Name->getName(); |
| unsigned Len = Name->getLength(); |
| |
| // Normalize the attribute name, __foo__ becomes foo. |
| if (Len > 4 && Str[0] == '_' && Str[1] == '_' && |
| Str[Len - 2] == '_' && Str[Len - 1] == '_') { |
| Str += 2; |
| Len -= 4; |
| } |
| |
| // FIXME: Hand generating this is neither smart nor efficient. |
| switch (Len) { |
| case 4: |
| if (!memcmp(Str, "weak", 4)) return AT_weak; |
| if (!memcmp(Str, "pure", 4)) return AT_pure; |
| if (!memcmp(Str, "mode", 4)) return AT_mode; |
| if (!memcmp(Str, "used", 4)) return AT_used; |
| break; |
| case 5: |
| if (!memcmp(Str, "alias", 5)) return AT_alias; |
| if (!memcmp(Str, "const", 5)) return AT_const; |
| break; |
| case 6: |
| if (!memcmp(Str, "packed", 6)) return AT_packed; |
| if (!memcmp(Str, "malloc", 6)) return IgnoredAttribute; // FIXME: noalias. |
| if (!memcmp(Str, "format", 6)) return AT_format; |
| if (!memcmp(Str, "unused", 6)) return AT_unused; |
| if (!memcmp(Str, "blocks", 6)) return AT_blocks; |
| break; |
| case 7: |
| if (!memcmp(Str, "aligned", 7)) return AT_aligned; |
| if (!memcmp(Str, "cleanup", 7)) return AT_cleanup; |
| if (!memcmp(Str, "nodebug", 7)) return AT_nodebug; |
| if (!memcmp(Str, "nonnull", 7)) return AT_nonnull; |
| if (!memcmp(Str, "nothrow", 7)) return AT_nothrow; |
| if (!memcmp(Str, "objc_gc", 7)) return AT_objc_gc; |
| if (!memcmp(Str, "regparm", 7)) return AT_regparm; |
| if (!memcmp(Str, "section", 7)) return AT_section; |
| if (!memcmp(Str, "stdcall", 7)) return AT_stdcall; |
| break; |
| case 8: |
| if (!memcmp(Str, "annotate", 8)) return AT_annotate; |
| if (!memcmp(Str, "noreturn", 8)) return AT_noreturn; |
| if (!memcmp(Str, "noinline", 8)) return AT_noinline; |
| if (!memcmp(Str, "fastcall", 8)) return AT_fastcall; |
| if (!memcmp(Str, "iboutlet", 8)) return AT_IBOutlet; |
| if (!memcmp(Str, "sentinel", 8)) return AT_sentinel; |
| if (!memcmp(Str, "NSObject", 8)) return AT_nsobject; |
| break; |
| case 9: |
| if (!memcmp(Str, "dllimport", 9)) return AT_dllimport; |
| if (!memcmp(Str, "dllexport", 9)) return AT_dllexport; |
| if (!memcmp(Str, "may_alias", 9)) return IgnoredAttribute; // FIXME: TBAA |
| break; |
| case 10: |
| if (!memcmp(Str, "deprecated", 10)) return AT_deprecated; |
| if (!memcmp(Str, "visibility", 10)) return AT_visibility; |
| if (!memcmp(Str, "destructor", 10)) return AT_destructor; |
| if (!memcmp(Str, "format_arg", 10)) |
| return IgnoredAttribute; // FIXME: printf format string checking. |
| if (!memcmp(Str, "gnu_inline", 10)) return AT_gnu_inline; |
| break; |
| case 11: |
| if (!memcmp(Str, "weak_import", 11)) return AT_weak_import; |
| if (!memcmp(Str, "vector_size", 11)) return AT_vector_size; |
| if (!memcmp(Str, "constructor", 11)) return AT_constructor; |
| if (!memcmp(Str, "unavailable", 11)) return AT_unavailable; |
| break; |
| case 12: |
| if (!memcmp(Str, "overloadable", 12)) return AT_overloadable; |
| break; |
| case 13: |
| if (!memcmp(Str, "address_space", 13)) return AT_address_space; |
| if (!memcmp(Str, "always_inline", 13)) return AT_always_inline; |
| break; |
| case 14: |
| if (!memcmp(Str, "objc_exception", 14)) return AT_objc_exception; |
| break; |
| case 15: |
| if (!memcmp(Str, "ext_vector_type", 15)) return AT_ext_vector_type; |
| break; |
| case 17: |
| if (!memcmp(Str, "transparent_union", 17)) return AT_transparent_union; |
| if (!memcmp(Str, "analyzer_noreturn", 17)) return AT_analyzer_noreturn; |
| break; |
| case 18: |
| if (!memcmp(Str, "warn_unused_result", 18)) return AT_warn_unused_result; |
| break; |
| case 19: |
| if (!memcmp(Str, "cf_ownership_retain", 19)) |
| return AT_cf_ownership_retain; |
| if (!memcmp(Str, "ns_ownership_retain", 19)) |
| return AT_ns_ownership_retain; |
| break; |
| case 20: |
| if (!memcmp(Str, "cf_ownership_release", 20)) |
| return AT_cf_ownership_release; |
| if (!memcmp(Str, "ns_ownership_release", 20)) |
| return AT_ns_ownership_release; |
| if (!memcmp(Str, "ns_ownership_returns", 20)) |
| return AT_ns_ownership_returns; |
| break; |
| case 22: |
| if (!memcmp(Str, "no_instrument_function", 22)) |
| return AT_no_instrument_function; |
| break; |
| case 29: |
| if (!memcmp(Str, "ns_ownership_make_collectable", 29)) |
| return AT_ns_ownership_make_collectable; |
| break; |
| } |
| return UnknownAttribute; |
| } |