aboutsummaryrefslogtreecommitdiff
path: root/CodeGen/CGExprAgg.cpp
diff options
context:
space:
mode:
authorDevang Patel <dpatel@apple.com>2007-10-26 17:44:44 +0000
committerDevang Patel <dpatel@apple.com>2007-10-26 17:44:44 +0000
commit636c3d04673da8c8605d7e45640a2ff7aec648f1 (patch)
treec233685701f337f92731e97e77e20908ab823fee /CodeGen/CGExprAgg.cpp
parent8e53e720b3d7c962e91138a130dbd5d6c2def0e5 (diff)
Codegen array initializers.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@43385 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'CodeGen/CGExprAgg.cpp')
-rw-r--r--CodeGen/CGExprAgg.cpp78
1 files changed, 77 insertions, 1 deletions
diff --git a/CodeGen/CGExprAgg.cpp b/CodeGen/CGExprAgg.cpp
index 743c2486d3..5f353edcf0 100644
--- a/CodeGen/CGExprAgg.cpp
+++ b/CodeGen/CGExprAgg.cpp
@@ -16,6 +16,7 @@
#include "clang/AST/AST.h"
#include "llvm/Constants.h"
#include "llvm/Function.h"
+#include "llvm/GlobalVariable.h"
#include "llvm/Support/Compiler.h"
using namespace clang;
using namespace CodeGen;
@@ -73,6 +74,7 @@ public:
void VisitConditionalOperator(const ConditionalOperator *CO);
+ void VisitInitListExpr(InitListExpr *E);
// case Expr::ChooseExprClass:
};
} // end anonymous namespace.
@@ -101,7 +103,8 @@ void AggExprEmitter::EmitAggregateCopy(llvm::Value *DestPtr,
llvm::Value *MemCpyOps[4] = {
DestPtr, SrcPtr,
- llvm::ConstantInt::get(IntPtr, TypeInfo.first),
+ // TypeInfo.first describes size in bits.
+ llvm::ConstantInt::get(IntPtr, TypeInfo.first/8),
llvm::ConstantInt::get(llvm::Type::Int32Ty, TypeInfo.second)
};
@@ -178,6 +181,79 @@ void AggExprEmitter::VisitConditionalOperator(const ConditionalOperator *E) {
CGF.EmitBlock(ContBlock);
}
+void AggExprEmitter::VisitInitListExpr(InitListExpr *E) {
+
+ unsigned NumInitElements = E->getNumInits();
+
+ assert ( E->getType()->isArrayType()
+ && "Only Array initializers are supported");
+
+ std::vector<llvm::Constant*> ArrayElts;
+ const llvm::PointerType *APType = cast<llvm::PointerType>(DestPtr->getType());
+ const llvm::ArrayType *AType = cast<llvm::ArrayType>(APType->getElementType());
+
+ // Copy initializer elements.
+ bool AllConstElements = true;
+ unsigned i = 0;
+ for (i = 0; i < NumInitElements; ++i) {
+ if (llvm::Constant *C = dyn_cast<llvm::Constant>(CGF.EmitScalarExpr(E->getInit(i))))
+ ArrayElts.push_back(C);
+ else {
+ AllConstElements = false;
+ break;
+ }
+ }
+
+ unsigned NumArrayElements = AType->getNumElements();
+ const llvm::Type *ElementType = CGF.ConvertType(E->getInit(0)->getType());
+
+ if (AllConstElements) {
+ // Initialize remaining array elements.
+ for (/*Do not initialize i*/; i < NumArrayElements; ++i)
+ ArrayElts.push_back(llvm::Constant::getNullValue(ElementType));
+
+ // Create global value to hold this array.
+ llvm::Constant *V = llvm::ConstantArray::get(AType, ArrayElts);
+ V = new llvm::GlobalVariable(V->getType(), true,
+ llvm::GlobalValue::InternalLinkage,
+ V, ".array",
+ &CGF.CGM.getModule());
+
+ EmitAggregateCopy(DestPtr, V , E->getType());
+ return;
+ }
+
+ // Emit indiviudal array element stores.
+ unsigned index = 0;
+ llvm::Value *NextVal = NULL;
+ llvm::Value *Idxs[] = {
+ llvm::Constant::getNullValue(llvm::Type::Int32Ty),
+ NULL
+ };
+
+ // Emit already seen constants initializers.
+ for (i = 0; i < ArrayElts.size(); i++) {
+ Idxs[1] = llvm::ConstantInt::get(llvm::Type::Int32Ty, index++);
+ NextVal = Builder.CreateGEP(DestPtr, Idxs, Idxs + 2, ".array");
+ Builder.CreateStore(ArrayElts[i], NextVal);
+ }
+
+ // Emit remaining initializers
+ for (/*Do not initizalize i*/; i < NumInitElements; ++i) {
+ Idxs[1] = llvm::ConstantInt::get(llvm::Type::Int32Ty, index++);
+ NextVal = Builder.CreateGEP(DestPtr, Idxs, Idxs + 2, ".array");
+ llvm::Value *V = CGF.EmitScalarExpr(E->getInit(i));
+ Builder.CreateStore(V, NextVal);
+ }
+
+ // Emit remaining default initializers
+ for (/*Do not initialize i*/; i < NumArrayElements; ++i) {
+ Idxs[1] = llvm::ConstantInt::get(llvm::Type::Int32Ty, index++);
+ NextVal = Builder.CreateGEP(DestPtr, Idxs, Idxs + 2, ".array");
+ Builder.CreateStore(llvm::Constant::getNullValue(ElementType), NextVal);
+ }
+}
+
//===----------------------------------------------------------------------===//
// Entry Points into this File
//===----------------------------------------------------------------------===//