mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-12-28 14:36:34 +00:00
fa995d7cc5
the section or the visibility from one global value to another: copyAttributesFrom. This is particularly useful for duplicating functions: previously this was done by explicitly copying each attribute in turn at each place where a new function was created out of an old one, with the result that obscure attributes were regularly forgotten (like the collector or the section). Hopefully now everything is uniform and nothing is forgotten. llvm-svn: 51567
164 lines
6.0 KiB
C++
164 lines
6.0 KiB
C++
//===-- llvm/GlobalVariable.h - GlobalVariable class ------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file contains the declaration of the GlobalVariable class, which
|
|
// represents a single global variable (or constant) in the VM.
|
|
//
|
|
// Global variables are constant pointers that refer to hunks of space that are
|
|
// allocated by either the VM, or by the linker in a static compiler. A global
|
|
// variable may have an intial value, which is copied into the executables .data
|
|
// area. Global Constants are required to have initializers.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_GLOBAL_VARIABLE_H
|
|
#define LLVM_GLOBAL_VARIABLE_H
|
|
|
|
#include "llvm/GlobalValue.h"
|
|
#include "llvm/OperandTraits.h"
|
|
|
|
namespace llvm {
|
|
|
|
class Module;
|
|
class Constant;
|
|
class PointerType;
|
|
template<typename ValueSubClass, typename ItemParentClass>
|
|
class SymbolTableListTraits;
|
|
|
|
class GlobalVariable : public GlobalValue {
|
|
friend class SymbolTableListTraits<GlobalVariable, Module>;
|
|
void *operator new(size_t, unsigned); // Do not implement
|
|
void operator=(const GlobalVariable &); // Do not implement
|
|
GlobalVariable(const GlobalVariable &); // Do not implement
|
|
|
|
void setParent(Module *parent);
|
|
|
|
GlobalVariable *Prev, *Next;
|
|
void setNext(GlobalVariable *N) { Next = N; }
|
|
void setPrev(GlobalVariable *N) { Prev = N; }
|
|
|
|
bool isConstantGlobal : 1; // Is this a global constant?
|
|
bool isThreadLocalSymbol : 1; // Is this symbol "Thread Local"?
|
|
|
|
public:
|
|
// allocate space for exactly one operand
|
|
void *operator new(size_t s) {
|
|
return User::operator new(s, 1);
|
|
}
|
|
/// GlobalVariable ctor - If a parent module is specified, the global is
|
|
/// automatically inserted into the end of the specified modules global list.
|
|
GlobalVariable(const Type *Ty, bool isConstant, LinkageTypes Linkage,
|
|
Constant *Initializer = 0, const std::string &Name = "",
|
|
Module *Parent = 0, bool ThreadLocal = false,
|
|
unsigned AddressSpace = 0);
|
|
/// GlobalVariable ctor - This creates a global and inserts it before the
|
|
/// specified other global.
|
|
GlobalVariable(const Type *Ty, bool isConstant, LinkageTypes Linkage,
|
|
Constant *Initializer, const std::string &Name,
|
|
GlobalVariable *InsertBefore, bool ThreadLocal = false,
|
|
unsigned AddressSpace = 0);
|
|
|
|
~GlobalVariable() {
|
|
NumOperands = 1; // FIXME: needed by operator delete
|
|
}
|
|
|
|
/// Provide fast operand accessors
|
|
DECLARE_TRANSPARENT_OPERAND_ACCESSORS(Value);
|
|
|
|
/// isDeclaration - Is this global variable lacking an initializer? If so,
|
|
/// the global variable is defined in some other translation unit, and is thus
|
|
/// only a declaration here.
|
|
virtual bool isDeclaration() const { return getNumOperands() == 0; }
|
|
|
|
/// hasInitializer - Unless a global variable isExternal(), it has an
|
|
/// initializer. The initializer for the global variable/constant is held by
|
|
/// Initializer if an initializer is specified.
|
|
///
|
|
inline bool hasInitializer() const { return !isDeclaration(); }
|
|
|
|
/// getInitializer - Return the initializer for this global variable. It is
|
|
/// illegal to call this method if the global is external, because we cannot
|
|
/// tell what the value is initialized to!
|
|
///
|
|
inline /*const FIXME*/ Constant *getInitializer() const {
|
|
assert(hasInitializer() && "GV doesn't have initializer!");
|
|
return static_cast<Constant*>(Op<0>().get());
|
|
}
|
|
inline Constant *getInitializer() {
|
|
assert(hasInitializer() && "GV doesn't have initializer!");
|
|
return static_cast<Constant*>(Op<0>().get());
|
|
}
|
|
inline void setInitializer(Constant *CPV) {
|
|
if (CPV == 0) {
|
|
if (hasInitializer()) {
|
|
Op<0>().set(0);
|
|
NumOperands = 0;
|
|
}
|
|
} else {
|
|
if (!hasInitializer())
|
|
NumOperands = 1;
|
|
Op<0>().set(CPV);
|
|
}
|
|
}
|
|
|
|
/// If the value is a global constant, its value is immutable throughout the
|
|
/// runtime execution of the program. Assigning a value into the constant
|
|
/// leads to undefined behavior.
|
|
///
|
|
bool isConstant() const { return isConstantGlobal; }
|
|
void setConstant(bool Val) { isConstantGlobal = Val; }
|
|
|
|
/// If the value is "Thread Local", its value isn't shared by the threads.
|
|
bool isThreadLocal() const { return isThreadLocalSymbol; }
|
|
void setThreadLocal(bool Val) { isThreadLocalSymbol = Val; }
|
|
|
|
/// copyAttributesFrom - copy all additional attributes (those not needed to
|
|
/// create a GlobalVariable) from the GlobalVariable Src to this one.
|
|
void copyAttributesFrom(const GlobalValue *Src);
|
|
|
|
/// removeFromParent - This method unlinks 'this' from the containing module,
|
|
/// but does not delete it.
|
|
///
|
|
void removeFromParent();
|
|
|
|
/// eraseFromParent - This method unlinks 'this' from the containing module
|
|
/// and deletes it.
|
|
///
|
|
void eraseFromParent();
|
|
|
|
/// Override Constant's implementation of this method so we can
|
|
/// replace constant initializers.
|
|
virtual void replaceUsesOfWithOnConstant(Value *From, Value *To, Use *U);
|
|
|
|
virtual void print(std::ostream &OS) const;
|
|
void print(std::ostream *OS) const { if (OS) print(*OS); }
|
|
|
|
// Methods for support type inquiry through isa, cast, and dyn_cast:
|
|
static inline bool classof(const GlobalVariable *) { return true; }
|
|
static inline bool classof(const Value *V) {
|
|
return V->getValueID() == Value::GlobalVariableVal;
|
|
}
|
|
private:
|
|
// getNext/Prev - Return the next or previous global variable in the list.
|
|
GlobalVariable *getNext() { return Next; }
|
|
const GlobalVariable *getNext() const { return Next; }
|
|
GlobalVariable *getPrev() { return Prev; }
|
|
const GlobalVariable *getPrev() const { return Prev; }
|
|
};
|
|
|
|
template <>
|
|
struct OperandTraits<GlobalVariable> : OptionalOperandTraits<> {
|
|
};
|
|
|
|
DEFINE_TRANSPARENT_OPERAND_ACCESSORS(GlobalVariable, Value)
|
|
|
|
} // End llvm namespace
|
|
|
|
#endif
|