1 //===-- AttributeImpl.h - Attribute Internals -------------------*- C++ -*-===//
3 // The LLVM Compiler Infrastructure
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
8 //===----------------------------------------------------------------------===//
11 /// \brief This file defines various helper methods and classes used by
12 /// LLVMContextImpl for creating and managing attributes.
14 //===----------------------------------------------------------------------===//
16 #ifndef LLVM_ATTRIBUTESIMPL_H
17 #define LLVM_ATTRIBUTESIMPL_H
19 #include "llvm/ADT/FoldingSet.h"
20 #include "llvm/IR/Attributes.h"
28 //===----------------------------------------------------------------------===//
30 /// \brief A set of classes that contain the kind and (optional) value of the
31 /// attribute object. There are three main categories: enum attribute entries,
32 /// represented by Attribute::AttrKind; alignment attribute entries; and string
33 /// attribute enties, which are for target-dependent attributes.
34 class AttributeEntry {
43 AttributeEntry(AttrEntryKind Kind)
45 virtual ~AttributeEntry() {}
47 unsigned getKindID() const { return KindID; }
49 static inline bool classof(const AttributeEntry *) { return true; }
52 class EnumAttributeEntry : public AttributeEntry {
53 Attribute::AttrKind Kind;
55 EnumAttributeEntry(Attribute::AttrKind Kind)
56 : AttributeEntry(EnumAttrEntry), Kind(Kind) {}
58 Attribute::AttrKind getEnumKind() const { return Kind; }
60 static inline bool classof(const AttributeEntry *AE) {
61 return AE->getKindID() == EnumAttrEntry;
63 static inline bool classof(const EnumAttributeEntry *) { return true; }
66 class AlignAttributeEntry : public AttributeEntry {
67 Attribute::AttrKind Kind;
70 AlignAttributeEntry(Attribute::AttrKind Kind, unsigned Align)
71 : AttributeEntry(AlignAttrEntry), Kind(Kind), Align(Align) {}
73 Attribute::AttrKind getEnumKind() const { return Kind; }
74 unsigned getAlignment() const { return Align; }
76 static inline bool classof(const AttributeEntry *AE) {
77 return AE->getKindID() == AlignAttrEntry;
79 static inline bool classof(const AlignAttributeEntry *) { return true; }
82 class StringAttributeEntry : public AttributeEntry {
86 StringAttributeEntry(StringRef Kind, StringRef Val = StringRef())
87 : AttributeEntry(StringAttrEntry), Kind(Kind), Val(Val) {}
89 StringRef getStringKind() const { return Kind; }
90 StringRef getStringValue() const { return Val; }
92 static inline bool classof(const AttributeEntry *AE) {
93 return AE->getKindID() == StringAttrEntry;
95 static inline bool classof(const StringAttributeEntry *) { return true; }
98 //===----------------------------------------------------------------------===//
100 /// \brief This class represents a single, uniqued attribute. That attribute
101 /// could be a single enum, a tuple, or a string.
102 class AttributeImpl : public FoldingSetNode {
103 LLVMContext &Context; ///< Global context for uniquing objects
105 AttributeEntry *Entry; ///< Holds the kind and value of the attribute
107 // AttributesImpl is uniqued, these should not be publicly available.
108 void operator=(const AttributeImpl &) LLVM_DELETED_FUNCTION;
109 AttributeImpl(const AttributeImpl &) LLVM_DELETED_FUNCTION;
111 AttributeImpl(LLVMContext &C, Attribute::AttrKind Kind);
112 AttributeImpl(LLVMContext &C, Attribute::AttrKind Kind, unsigned Align);
113 AttributeImpl(LLVMContext &C, StringRef Kind, StringRef Val = StringRef());
116 LLVMContext &getContext() { return Context; }
118 bool isEnumAttribute() const;
119 bool isAlignAttribute() const;
120 bool isStringAttribute() const;
122 bool hasAttribute(Attribute::AttrKind A) const;
123 bool hasAttribute(StringRef Kind) const;
125 Attribute::AttrKind getKindAsEnum() const;
126 uint64_t getValueAsInt() const;
128 StringRef getKindAsString() const;
129 StringRef getValueAsString() const;
131 /// \brief Used when sorting the attributes.
132 bool operator<(const AttributeImpl &AI) const;
134 void Profile(FoldingSetNodeID &ID) const {
135 if (isEnumAttribute())
136 Profile(ID, getKindAsEnum(), 0);
137 else if (isAlignAttribute())
138 Profile(ID, getKindAsEnum(), getValueAsInt());
140 Profile(ID, getKindAsString(), getValueAsString());
142 static void Profile(FoldingSetNodeID &ID, Attribute::AttrKind Kind,
145 if (Val) ID.AddInteger(Val);
147 static void Profile(FoldingSetNodeID &ID, StringRef Kind, StringRef Values) {
149 if (!Values.empty()) ID.AddString(Values);
152 // FIXME: Remove this!
153 static uint64_t getAttrMask(Attribute::AttrKind Val);
156 //===----------------------------------------------------------------------===//
158 /// \brief This class represents a group of attributes that apply to one
159 /// element: function, return type, or parameter.
160 class AttributeSetNode : public FoldingSetNode {
161 SmallVector<Attribute, 4> AttrList;
163 AttributeSetNode(ArrayRef<Attribute> Attrs)
164 : AttrList(Attrs.begin(), Attrs.end()) {}
166 // AttributesSetNode is uniqued, these should not be publicly available.
167 void operator=(const AttributeSetNode &) LLVM_DELETED_FUNCTION;
168 AttributeSetNode(const AttributeSetNode &) LLVM_DELETED_FUNCTION;
170 static AttributeSetNode *get(LLVMContext &C, ArrayRef<Attribute> Attrs);
172 bool hasAttribute(Attribute::AttrKind Kind) const;
173 bool hasAttribute(StringRef Kind) const;
174 bool hasAttributes() const { return !AttrList.empty(); }
176 Attribute getAttribute(Attribute::AttrKind Kind) const;
177 Attribute getAttribute(StringRef Kind) const;
179 unsigned getAlignment() const;
180 unsigned getStackAlignment() const;
181 std::string getAsString(bool InAttrGrp) const;
183 typedef SmallVectorImpl<Attribute>::iterator iterator;
184 typedef SmallVectorImpl<Attribute>::const_iterator const_iterator;
186 iterator begin() { return AttrList.begin(); }
187 iterator end() { return AttrList.end(); }
189 const_iterator begin() const { return AttrList.begin(); }
190 const_iterator end() const { return AttrList.end(); }
192 void Profile(FoldingSetNodeID &ID) const {
193 Profile(ID, AttrList);
195 static void Profile(FoldingSetNodeID &ID, ArrayRef<Attribute> AttrList) {
196 for (unsigned I = 0, E = AttrList.size(); I != E; ++I)
197 AttrList[I].Profile(ID);
201 //===----------------------------------------------------------------------===//
203 /// \brief This class represents a set of attributes that apply to the function,
204 /// return type, and parameters.
205 class AttributeSetImpl : public FoldingSetNode {
206 friend class AttributeSet;
208 LLVMContext &Context;
210 typedef std::pair<unsigned, AttributeSetNode*> IndexAttrPair;
211 SmallVector<IndexAttrPair, 4> AttrNodes;
213 // AttributesSet is uniqued, these should not be publicly available.
214 void operator=(const AttributeSetImpl &) LLVM_DELETED_FUNCTION;
215 AttributeSetImpl(const AttributeSetImpl &) LLVM_DELETED_FUNCTION;
217 AttributeSetImpl(LLVMContext &C,
218 ArrayRef<std::pair<unsigned, AttributeSetNode*> > attrs)
219 : Context(C), AttrNodes(attrs.begin(), attrs.end()) {}
221 /// \brief Get the context that created this AttributeSetImpl.
222 LLVMContext &getContext() { return Context; }
224 /// \brief Return the number of attributes this AttributeSet contains.
225 unsigned getNumAttributes() const { return AttrNodes.size(); }
227 /// \brief Get the index of the given "slot" in the AttrNodes list. This index
228 /// is the index of the return, parameter, or function object that the
229 /// attributes are applied to, not the index into the AttrNodes list where the
230 /// attributes reside.
231 unsigned getSlotIndex(unsigned Slot) const {
232 return AttrNodes[Slot].first;
235 /// \brief Retrieve the attributes for the given "slot" in the AttrNode list.
236 /// \p Slot is an index into the AttrNodes list, not the index of the return /
237 /// parameter/ function which the attributes apply to.
238 AttributeSet getSlotAttributes(unsigned Slot) const {
239 return AttributeSet::get(Context, AttrNodes[Slot]);
242 /// \brief Retrieve the attribute set node for the given "slot" in the
244 AttributeSetNode *getSlotNode(unsigned Slot) const {
245 return AttrNodes[Slot].second;
248 typedef AttributeSetNode::iterator iterator;
249 typedef AttributeSetNode::const_iterator const_iterator;
251 iterator begin(unsigned Slot)
252 { return AttrNodes[Slot].second->begin(); }
253 iterator end(unsigned Slot)
254 { return AttrNodes[Slot].second->end(); }
256 const_iterator begin(unsigned Slot) const
257 { return AttrNodes[Slot].second->begin(); }
258 const_iterator end(unsigned Slot) const
259 { return AttrNodes[Slot].second->end(); }
261 void Profile(FoldingSetNodeID &ID) const {
262 Profile(ID, AttrNodes);
264 static void Profile(FoldingSetNodeID &ID,
265 ArrayRef<std::pair<unsigned, AttributeSetNode*> > Nodes) {
266 for (unsigned i = 0, e = Nodes.size(); i != e; ++i) {
267 ID.AddInteger(Nodes[i].first);
268 ID.AddPointer(Nodes[i].second);
272 // FIXME: This atrocity is temporary.
273 uint64_t Raw(unsigned Index) const;
276 } // end llvm namespace