//===-- llvm/OperandTraits.h - OperandTraits class definition ---*- 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 traits classes that are handy for enforcing the correct // layout of various User subclasses. It also provides the means for accessing // the operands in the most efficient manner. // #ifndef LLVM_OPERAND_TRAITS_H #define LLVM_OPERAND_TRAITS_H #include "llvm/User.h" namespace llvm { //===----------------------------------------------------------------------===// // FixedNumOperand Trait Class //===----------------------------------------------------------------------===// /// FixedNumOperandTraits - determine the allocation regime of the Use array /// when it is a prefix to the User object, and the number of Use objects is /// known at compile time. template struct FixedNumOperandTraits { static Use *op_begin(SubClass* U) { return reinterpret_cast(U) - ARITY; } static Use *op_end(SubClass* U) { return reinterpret_cast(U); } static unsigned operands(const User*) { return ARITY; } }; //===----------------------------------------------------------------------===// // OptionalOperand Trait Class //===----------------------------------------------------------------------===// /// OptionalOperandTraits - when the number of operands may change at runtime. /// Naturally it may only decrease, because the allocations may not change. template struct OptionalOperandTraits : public FixedNumOperandTraits { static unsigned operands(const User *U) { return U->getNumOperands(); } }; //===----------------------------------------------------------------------===// // VariadicOperand Trait Class //===----------------------------------------------------------------------===// /// VariadicOperandTraits - determine the allocation regime of the Use array /// when it is a prefix to the User object, and the number of Use objects is /// only known at allocation time. template struct VariadicOperandTraits { static Use *op_begin(SubClass* U) { return reinterpret_cast(U) - static_cast(U)->getNumOperands(); } static Use *op_end(SubClass* U) { return reinterpret_cast(U); } static unsigned operands(const User *U) { return U->getNumOperands(); } }; //===----------------------------------------------------------------------===// // HungoffOperand Trait Class //===----------------------------------------------------------------------===// /// HungoffOperandTraits - determine the allocation regime of the Use array /// when it is not a prefix to the User object, but allocated at an unrelated /// heap address. /// Assumes that the User subclass that is determined by this traits class /// has an OperandList member of type User::op_iterator. [Note: this is now /// trivially satisfied, because User has that member for historic reasons.] /// /// This is the traits class that is needed when the Use array must be /// resizable. template struct HungoffOperandTraits { static Use *op_begin(User* U) { return U->OperandList; } static Use *op_end(User* U) { return U->OperandList + U->getNumOperands(); } static unsigned operands(const User *U) { return U->getNumOperands(); } }; /// Macro for generating in-class operand accessor declarations. /// It should only be called in the public section of the interface. /// #define DECLARE_TRANSPARENT_OPERAND_ACCESSORS(VALUECLASS) \ public: \ inline VALUECLASS *getOperand(unsigned) const; \ inline void setOperand(unsigned, VALUECLASS*); \ inline op_iterator op_begin(); \ inline const_op_iterator op_begin() const; \ inline op_iterator op_end(); \ inline const_op_iterator op_end() const; \ protected: \ template inline Use &Op(); \ template inline const Use &Op() const; \ public: \ inline unsigned getNumOperands() const /// Macro for generating out-of-class operand accessor definitions #define DEFINE_TRANSPARENT_OPERAND_ACCESSORS(CLASS, VALUECLASS) \ CLASS::op_iterator CLASS::op_begin() { \ return OperandTraits::op_begin(this); \ } \ CLASS::const_op_iterator CLASS::op_begin() const { \ return OperandTraits::op_begin(const_cast(this)); \ } \ CLASS::op_iterator CLASS::op_end() { \ return OperandTraits::op_end(this); \ } \ CLASS::const_op_iterator CLASS::op_end() const { \ return OperandTraits::op_end(const_cast(this)); \ } \ VALUECLASS *CLASS::getOperand(unsigned i_nocapture) const { \ assert(i_nocapture < OperandTraits::operands(this) \ && "getOperand() out of range!"); \ return cast_or_null( \ OperandTraits::op_begin(const_cast(this))[i_nocapture].get()); \ } \ void CLASS::setOperand(unsigned i_nocapture, VALUECLASS *Val_nocapture) { \ assert(i_nocapture < OperandTraits::operands(this) \ && "setOperand() out of range!"); \ OperandTraits::op_begin(this)[i_nocapture] = Val_nocapture; \ } \ unsigned CLASS::getNumOperands() const { \ return OperandTraits::operands(this); \ } \ template Use &CLASS::Op() { \ return this->OpFrom(this); \ } \ template const Use &CLASS::Op() const { \ return this->OpFrom(this); \ } } // End llvm namespace #endif