Index: include/llvm/IR/Argument.h =================================================================== --- include/llvm/IR/Argument.h +++ include/llvm/IR/Argument.h @@ -121,6 +121,13 @@ /// \brief Add a Attribute to an argument. void addAttr(AttributeSet AS); + /// \brief Return the attribute object for the given attribute kind. + Attribute getAttr(StringRef Kind) const; + + /// \brief Return true if this argument has the attribute of the + /// specified kind. + bool hasAttr(StringRef Kind) const; + /// \brief Remove a Attribute from an argument. void removeAttr(AttributeSet AS); Index: include/llvm/IR/Function.h =================================================================== --- include/llvm/IR/Function.h +++ include/llvm/IR/Function.h @@ -274,6 +274,16 @@ return AttributeSets.getDereferenceableOrNullBytes(i); } + /// \brief Return the attribute object that exists at the given index. + Attribute getAttribute(unsigned i, StringRef Kind) const { + return AttributeSets.getAttribute(i, Kind); + } + + /// \brief Return true if attribute exists at the given index. + bool hasAttribute(unsigned i, StringRef Kind) const { + return AttributeSets.hasAttribute(i, Kind); + } + /// @brief Determine if the function does not access memory. bool doesNotAccessMemory() const { return AttributeSets.hasAttribute(AttributeSet::FunctionIndex, Index: lib/AsmParser/LLParser.h =================================================================== --- lib/AsmParser/LLParser.h +++ lib/AsmParser/LLParser.h @@ -210,6 +210,8 @@ return ParseUInt64(Val); } + bool ParseStringAttribute(AttrBuilder &B); + bool ParseTLSModel(GlobalVariable::ThreadLocalMode &TLM); bool ParseOptionalThreadLocal(GlobalVariable::ThreadLocalMode &TLM); bool parseOptionalUnnamedAddr(bool &UnnamedAddr) { Index: lib/AsmParser/LLParser.cpp =================================================================== --- lib/AsmParser/LLParser.cpp +++ lib/AsmParser/LLParser.cpp @@ -903,14 +903,8 @@ } // Target-dependent attributes: case lltok::StringConstant: { - std::string Attr = Lex.getStrVal(); - Lex.Lex(); - std::string Val; - if (EatIfPresent(lltok::equal) && - ParseStringConstant(Val)) + if (ParseStringAttribute(B)) return true; - - B.addAttribute(Attr, Val); continue; } @@ -1210,6 +1204,19 @@ ParseToken(lltok::rparen, "expected ')' in address space"); } +/// ParseStringAttribute +/// := StringConstant +/// := StringConstant '=' StringConstant +bool LLParser::ParseStringAttribute(AttrBuilder &B) { + std::string Attr = Lex.getStrVal(); + Lex.Lex(); + std::string Val; + if (EatIfPresent(lltok::equal) && ParseStringConstant(Val)) + return true; + B.addAttribute(Attr, Val); + return false; +} + /// ParseOptionalParamAttrs - Parse a potentially empty list of parameter attributes. bool LLParser::ParseOptionalParamAttrs(AttrBuilder &B) { bool HaveError = false; @@ -1221,6 +1228,11 @@ switch (Token) { default: // End of attributes. return HaveError; + case lltok::StringConstant: { + if (ParseStringAttribute(B)) + return true; + continue; + } case lltok::kw_align: { unsigned Alignment; if (ParseOptionalAlignment(Alignment)) @@ -1301,6 +1313,11 @@ switch (Token) { default: // End of attributes. return HaveError; + case lltok::StringConstant: { + if (ParseStringAttribute(B)) + return true; + continue; + } case lltok::kw_dereferenceable: { uint64_t Bytes; if (ParseOptionalDerefAttrBytes(lltok::kw_dereferenceable, Bytes)) Index: lib/IR/Function.cpp =================================================================== --- lib/IR/Function.cpp +++ lib/IR/Function.cpp @@ -198,6 +198,14 @@ getArgNo() + 1, B)); } +Attribute Argument::getAttr(StringRef Kind) const { + return getParent()->getAttribute(getArgNo()+1, Kind); +} + +bool Argument::hasAttr(StringRef Kind) const { + return getParent()->hasAttribute(getArgNo()+1, Kind); +} + /// removeAttr - Remove attributes from an argument. void Argument::removeAttr(AttributeSet AS) { assert(AS.getNumSlots() <= 1 && Index: test/Bitcode/attributes.ll =================================================================== --- test/Bitcode/attributes.ll +++ test/Bitcode/attributes.ll @@ -256,6 +256,16 @@ ret void } +; CHECK: define "string_attribute" void @f44(i32 "string_attribute") +define "string_attribute" void @f44(i32 "string_attribute") { + ret void +} + +; CHECK: define "string_attribute_with_value"="value" void @f45(i32 "string_attribute_with_value"="value") +define "string_attribute_with_value"="value" void @f45(i32 "string_attribute_with_value"="value") { + ret void +} + ; CHECK: attributes #0 = { noreturn } ; CHECK: attributes #1 = { nounwind } ; CHECK: attributes #2 = { readnone }