]> granicus.if.org Git - clang/commitdiff
Handle simple struct member expr.
authorDevang Patel <dpatel@apple.com>
Tue, 23 Oct 2007 20:28:39 +0000 (20:28 +0000)
committerDevang Patel <dpatel@apple.com>
Tue, 23 Oct 2007 20:28:39 +0000 (20:28 +0000)
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@43258 91177308-0d34-0410-b5e6-96231b3b80d8

CodeGen/CGExpr.cpp
test/CodeGen/struct.c [new file with mode: 0644]

index 7fbd1f17aa3b4bbd6c2d8643422bf36e661bbb96..c57b38d4076cde0ff2517a5f7d5620eb6134f276 100644 (file)
@@ -101,6 +101,7 @@ LValue CodeGenFunction::EmitLValue(const Expr *E) {
     return EmitArraySubscriptExpr(cast<ArraySubscriptExpr>(E));
   case Expr::OCUVectorElementExprClass:
     return EmitOCUVectorElementExpr(cast<OCUVectorElementExpr>(E));
+  case Expr::MemberExprClass: return EmitMemberExpr(cast<MemberExpr>(E));
   }
 }
 
@@ -381,6 +382,36 @@ EmitOCUVectorElementExpr(const OCUVectorElementExpr *E) {
                                   E->getEncodedElementAccess());
 }
 
+LValue CodeGenFunction::EmitMemberExpr(const MemberExpr *E) {
+
+  // FIXME: Handle union members.
+  if (E->getBase()->getType()->getAsUnionType()) {
+    fprintf(stderr, "Unimplemented lvalue expr!\n");
+    E->dump(getContext().SourceMgr);
+    llvm::Type *Ty = llvm::PointerType::get(ConvertType(E->getType()));
+    return LValue::MakeAddr(llvm::UndefValue::get(Ty));
+  }
+  
+  LValue BaseLV = EmitLValue(E->getBase());
+  llvm::Value *BaseValue = BaseLV.getAddress();
+
+  FieldDecl *Field = E->getMemberDecl();
+  unsigned idx = CGM.getTypes().getLLVMFieldNo(Field);
+  llvm::Value *Idxs[2] = { llvm::Constant::getNullValue(llvm::Type::Int32Ty), 
+                          llvm::ConstantInt::get(llvm::Type::Int32Ty, idx) };
+  if (E->isArrow()) {
+    QualType PTy = cast<PointerType>(E->getBase()->getType())->getPointeeType();
+    BaseValue =  Builder.CreateBitCast(BaseValue, 
+                                      llvm::PointerType::get(ConvertType(PTy)),
+                                      "tmp");
+  }
+
+  return LValue::MakeAddr(Builder.CreateGEP(BaseValue,Idxs, Idxs + 2, "tmp"));
+  
+  // FIXME: If record field does not have one to one match with llvm::StructType
+  // field then apply appropriate masks to select only member field bits.
+}
+
 //===--------------------------------------------------------------------===//
 //                             Expression Emission
 //===--------------------------------------------------------------------===//
diff --git a/test/CodeGen/struct.c b/test/CodeGen/struct.c
new file mode 100644 (file)
index 0000000..a3753d3
--- /dev/null
@@ -0,0 +1,39 @@
+// RUN: clang %s -emit-llvm
+
+struct  {
+  int x;
+  int y;
+} point;
+
+void fn1() {
+  point.x = 42;
+}
+
+/* Nested member */
+struct  {
+  struct {
+    int a;
+    int b;
+  } p1;
+} point2;
+
+void fn2() {
+  point2.p1.a = 42;
+}
+
+/* Indirect reference */
+typedef struct __sf {
+ unsigned char *c;
+ short flags;
+} F;
+
+typedef struct __sf2 {
+  F *ff;
+} F2;
+
+int fn3(F2 *c) {
+  if (c->ff->c >= 0)
+    return 1;
+  else
+    return 0;
+}