From fad187ee31b03d00d651f3d2981aabca6c693e79 Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Wed, 23 Aug 2023 11:22:31 -0700 Subject: [PATCH 01/14] Modernize use of LLVM pointer types (part 1). --- src/tcompiler.cpp | 24 +++++++++++++----------- 1 file changed, 13 insertions(+), 11 deletions(-) diff --git a/src/tcompiler.cpp b/src/tcompiler.cpp index 3449e888..f2c435b7 100644 --- a/src/tcompiler.cpp +++ b/src/tcompiler.cpp @@ -736,10 +736,9 @@ static AllocaInst *CreateAlloca(IRBuilder<> *B, Type *Ty, Value *ArraySize = 0, return TmpB.CreateAlloca(Ty, ArraySize, Name); } -static Value *CreateConstGEP2_32(IRBuilder<> *B, Value *Ptr, unsigned Idx0, - unsigned Idx1) { - return B->CreateConstGEP2_32(Ptr->getType()->getPointerElementType(), Ptr, Idx0, - Idx1); +static Value *CreateConstGEP2_32(IRBuilder<> *B, Value *Ptr, Type *ValueType, + unsigned Idx0, unsigned Idx1) { + return B->CreateConstGEP2_32(ValueType, Ptr, Idx0, Idx1); } // functions that handle the details of the x86_64 ABI (this really should be handled by @@ -1285,7 +1284,7 @@ struct CCallingConv { int N = st->getNumElements(); for (int j = 0; j < N; j++) { Type *elt_type = st->getElementType(j); - EmitEntryAggReg(B, CreateConstGEP2_32(B, dest, 0, j), elt_type, ai); + EmitEntryAggReg(B, CreateConstGEP2_32(B, dest, st, 0, j), elt_type, ai); } } else { B->CreateStore(&*ai, dest); @@ -1354,7 +1353,7 @@ struct CCallingConv { Type *result_type = type; if (info->returntype.GetNumberOfTypesInParamList() == 1) { do { - result = CreateConstGEP2_32(B, result, 0, 0); + result = CreateConstGEP2_32(B, result, type, 0, 0); result_type = type->getElementType(0); } while ((type = dyn_cast(result_type))); } @@ -1378,7 +1377,7 @@ struct CCallingConv { int N = st->getNumElements(); for (int j = 0; j < N; j++) { Type *elt_type = st->getElementType(j); - EmitCallAggReg(B, CreateConstGEP2_32(B, value, 0, j), elt_type, + EmitCallAggReg(B, CreateConstGEP2_32(B, value, st, 0, j), elt_type, arguments); } } else { @@ -1454,7 +1453,7 @@ struct CCallingConv { Value *casted = B->CreateBitCast(aggregate, Ptr(type, as)); if (info.returntype.GetNumberOfTypesInParamList() == 1) { do { - casted = CreateConstGEP2_32(B, casted, 0, 0); + casted = CreateConstGEP2_32(B, casted, type, 0, 0); } while ((type = dyn_cast(type->getElementType(0)))); } if (info.returntype.GetNumberOfTypesInParamList() > 0) @@ -2283,8 +2282,9 @@ struct FunctionEmitter { return 0; } Value *emitArrayToPointer(Obj *exp) { + TType *t = typeOfValue(exp); Value *v = emitAddressOf(exp); - return CreateConstGEP2_32(B, v, 0, 0); + return CreateConstGEP2_32(B, v, t->type, 0, 0); } Type *getPrimitiveType(TType *t) { if (t->type->isVectorTy()) @@ -2364,7 +2364,9 @@ struct FunctionEmitter { int allocindex = entry.number("allocation"); - Value *addr = CreateConstGEP2_32(B, structPtr, 0, allocindex); + Value *addr = CreateConstGEP2_32(B, structPtr, + structPtr->getType()->getPointerElementType(), 0, + allocindex); // in three cases the type of the value in the struct does not match the expected // type returned // 1. if it is a union then the llvm struct will have some buffer space to hold @@ -3142,7 +3144,7 @@ struct FunctionEmitter { std::vector values; emitExpressionList(expressions, true, &values); for (size_t i = 0; i < values.size(); i++) { - Value *addr = CreateConstGEP2_32(B, result, 0, i); + Value *addr = CreateConstGEP2_32(B, result, ttype, 0, i); B->CreateStore(values[i], addr); } return B->CreateLoad(result->getType()->getPointerElementType(), result); From ed3be69b2425c4519cadce4b95e6f537825e6471 Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Wed, 23 Aug 2023 14:05:24 -0700 Subject: [PATCH 02/14] Fix up some more cases. --- src/tcompiler.cpp | 10 ++++------ 1 file changed, 4 insertions(+), 6 deletions(-) diff --git a/src/tcompiler.cpp b/src/tcompiler.cpp index f2c435b7..0628fe12 100644 --- a/src/tcompiler.cpp +++ b/src/tcompiler.cpp @@ -1381,8 +1381,7 @@ struct CCallingConv { arguments); } } else { - arguments.push_back( - B->CreateLoad(value->getType()->getPointerElementType(), value)); + arguments.push_back(B->CreateLoad(param_type, value)); } } @@ -1467,8 +1466,7 @@ struct CCallingConv { } else { assert(!"unhandled argument kind"); } - return B->CreateLoad(aggregate->getType()->getPointerElementType(), - aggregate); + return B->CreateLoad(info.returntype.type->type, aggregate); } } void GatherArgumentsAggReg(Type *type, std::vector &arguments) { @@ -3147,7 +3145,7 @@ struct FunctionEmitter { Value *addr = CreateConstGEP2_32(B, result, ttype, 0, i); B->CreateStore(values[i], addr); } - return B->CreateLoad(result->getType()->getPointerElementType(), result); + return B->CreateLoad(ttype, result); } void emitStmtList(Obj *stmts) { int NS = stmts->size(); @@ -3318,7 +3316,7 @@ struct FunctionEmitter { BasicBlock *cond = createAndInsertBB("forcond"); B->CreateBr(cond); setInsertBlock(cond); - Value *v = B->CreateLoad(vp->getType()->getPointerElementType(), vp); + Value *v = B->CreateLoad(t->type, vp); Value *c = B->CreateOr(B->CreateAnd(emitCompare(T_lt, t, v, limitv), emitCompare(T_gt, t, stepv, zero)), B->CreateAnd(emitCompare(T_gt, t, v, limitv), From 64918ef49549d8522643fef1751cf068be3be31e Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Thu, 24 Aug 2023 10:49:50 -0700 Subject: [PATCH 03/14] A few more cases. --- src/tcompiler.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/tcompiler.cpp b/src/tcompiler.cpp index 0628fe12..94283e89 100644 --- a/src/tcompiler.cpp +++ b/src/tcompiler.cpp @@ -2363,7 +2363,7 @@ struct FunctionEmitter { int allocindex = entry.number("allocation"); Value *addr = CreateConstGEP2_32(B, structPtr, - structPtr->getType()->getPointerElementType(), 0, + getType(structType)->type, 0, allocindex); // in three cases the type of the value in the struct does not match the expected // type returned @@ -2456,7 +2456,7 @@ struct FunctionEmitter { exp->obj("type", &type); Ty->EnsureTypeIsComplete(&type); Type *ttype = getType(&type)->type; - raw = B->CreateLoad(raw->getType()->getPointerElementType(), raw); + raw = B->CreateLoad(ttype, raw); } return raw; } From 6220ea23ae19db7f435514b1efa21381d0cd35ed Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Thu, 24 Aug 2023 11:54:38 -0700 Subject: [PATCH 04/14] Format. --- src/tcompiler.cpp | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/tcompiler.cpp b/src/tcompiler.cpp index 94283e89..a7d84d4e 100644 --- a/src/tcompiler.cpp +++ b/src/tcompiler.cpp @@ -2362,8 +2362,7 @@ struct FunctionEmitter { int allocindex = entry.number("allocation"); - Value *addr = CreateConstGEP2_32(B, structPtr, - getType(structType)->type, 0, + Value *addr = CreateConstGEP2_32(B, structPtr, getType(structType)->type, 0, allocindex); // in three cases the type of the value in the struct does not match the expected // type returned From a0e42904b64dd901d550d6f583b2d555d39d14b9 Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Thu, 24 Aug 2023 12:10:33 -0700 Subject: [PATCH 05/14] Fix one more. --- src/tcompiler.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/tcompiler.cpp b/src/tcompiler.cpp index a7d84d4e..9eafe700 100644 --- a/src/tcompiler.cpp +++ b/src/tcompiler.cpp @@ -2689,7 +2689,7 @@ struct FunctionEmitter { // structvariable and perform any casts necessary B->CreateStore(in, oe); } - return B->CreateLoad(output->getType()->getPointerElementType(), output); + return B->CreateLoad(toT->type, output); } break; case T_cast: { Obj a; From c7acd28c35221d862f29e50d9d60957af216124c Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Thu, 24 Aug 2023 13:49:27 -0700 Subject: [PATCH 06/14] Fix one more. --- src/tcompiler.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/tcompiler.cpp b/src/tcompiler.cpp index 9eafe700..8bf13837 100644 --- a/src/tcompiler.cpp +++ b/src/tcompiler.cpp @@ -2792,7 +2792,7 @@ struct FunctionEmitter { Ty->EnsureTypeIsComplete(&type); Type *ttype = getType(&type)->type; Value *v = emitExp(&addr); - LoadInst *l = B->CreateLoad(v->getType()->getPointerElementType(), v); + LoadInst *l = B->CreateLoad(ttype, v); if (attr.hasfield("alignment")) { int alignment = attr.number("alignment"); l->setAlignment(Align(alignment)); From c491a94069b9260db114e1970dc030e80555b2d5 Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Thu, 24 Aug 2023 13:55:02 -0700 Subject: [PATCH 07/14] Fix one more. --- src/tllvmutil.cpp | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/tllvmutil.cpp b/src/tllvmutil.cpp index 966fcfeb..c6b682ea 100644 --- a/src/tllvmutil.cpp +++ b/src/tllvmutil.cpp @@ -201,10 +201,10 @@ struct CopyConnectedComponent : public ValueMaterializer { } else if (GlobalVariable *GV = dyn_cast(V)) { GlobalVariable *newGV = dest->getGlobalVariable(GV->getName(), true); if (!newGV || needsFreshlyNamedConstant(GV, newGV)) { - newGV = new GlobalVariable( - *dest, GV->getType()->getPointerElementType(), GV->isConstant(), - GV->getLinkage(), NULL, GV->getName(), NULL, - GlobalVariable::NotThreadLocal, GV->getType()->getAddressSpace()); + newGV = new GlobalVariable(*dest, GV->getValueType(), GV->isConstant(), + GV->getLinkage(), NULL, GV->getName(), NULL, + GlobalVariable::NotThreadLocal, + GV->getType()->getAddressSpace()); newGV->copyAttributesFrom(GV); // copyAttributesFrom does not copy comdats newGV->setComdat(GV->getComdat()); From d2f203aa6ba27f0c8b0aa2a266d99e6bd2e22123 Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Thu, 24 Aug 2023 16:27:04 -0700 Subject: [PATCH 08/14] Fix a pointer case. --- src/tcompiler.cpp | 16 ++++++++++++++-- 1 file changed, 14 insertions(+), 2 deletions(-) diff --git a/src/tcompiler.cpp b/src/tcompiler.cpp index 8bf13837..95ac05ef 100644 --- a/src/tcompiler.cpp +++ b/src/tcompiler.cpp @@ -723,6 +723,15 @@ class Types { EnsureTypeIsComplete(&objTy); } // otherwise it is niltype and already complete } + static void PointsToType(Obj *ptrTy, Obj *result) { + if (ptrTy->kind("kind") == T_pointer) { + ptrTy->obj("type", result); + } else { + printf("unexpected kind = %d, %s\n", ptrTy->kind("kind"), + tkindtostr(ptrTy->kind("kind"))); + assert(false); + } + } }; // helper function to alloca at the beginning of function @@ -2568,13 +2577,16 @@ struct FunctionEmitter { Value *result = B->CreateExtractElement(valueExp, idxExp); return result; } else { + assert(aggType->type->isPointerTy()); + Obj objType; + Types::PointsToType(&aggTypeO, &objType); + idxExp = emitIndex(typeOfValue(&idx), 64, idxExp); // otherwise we have a pointer access which will use a GEP instruction std::vector idxs; Ty->EnsurePointsToCompleteType(&aggTypeO); Value *result = - B->CreateGEP(valueExp->getType()->getPointerElementType(), - valueExp, idxExp); + B->CreateGEP(getType(&objType)->type, valueExp, idxExp); if (!exp->boolean("lvalue")) result = B->CreateLoad(result->getType()->getPointerElementType(), result); From 203b3147c24e2d01d88eb3d8d3910b8204999c7c Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Thu, 24 Aug 2023 16:32:59 -0700 Subject: [PATCH 09/14] Fix some more pointer cases. --- src/tcompiler.cpp | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/src/tcompiler.cpp b/src/tcompiler.cpp index 95ac05ef..67403199 100644 --- a/src/tcompiler.cpp +++ b/src/tcompiler.cpp @@ -2164,15 +2164,17 @@ struct FunctionEmitter { ttype.issigned = ftype->issigned; return emitPrimitiveCast(ftype, &ttype, number); } - Value *emitPointerArith(T_Kind kind, Value *pointer, TType *numTy, Value *number) { + Value *emitPointerArith(T_Kind kind, Obj *ptrTy, Value *pointer, TType *numTy, + Value *number) { + Obj objTy; + Types::PointsToType(ptrTy, &objTy); + number = emitIndex(numTy, 64, number); if (kind == T_add) { - return B->CreateGEP(pointer->getType()->getPointerElementType(), pointer, - number); + return B->CreateGEP(getType(&objTy)->type, pointer, number); } else if (kind == T_sub) { Value *numNeg = B->CreateNeg(number); - return B->CreateGEP(pointer->getType()->getPointerElementType(), pointer, - numNeg); + return B->CreateGEP(getType(&objTy)->type, pointer, numNeg); } else { assert(!"unexpected pointer arith"); return NULL; @@ -2220,7 +2222,7 @@ struct FunctionEmitter { return emitPointerSub(t, a, b); } else { assert(bt->type->isIntegerTy()); - return emitPointerArith(kind, a, bt, b); + return emitPointerArith(kind, &aot, a, bt, b); } } From 44a526c9993cfb2b00bdcd1b3264d4d96c6270e1 Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Thu, 24 Aug 2023 16:35:43 -0700 Subject: [PATCH 10/14] Fix another pointer case. --- src/tcompiler.cpp | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/src/tcompiler.cpp b/src/tcompiler.cpp index 67403199..9e421197 100644 --- a/src/tcompiler.cpp +++ b/src/tcompiler.cpp @@ -2180,10 +2180,12 @@ struct FunctionEmitter { return NULL; } } - Value *emitPointerSub(TType *t, Value *a, Value *b) { + Value *emitPointerSub(Obj *ptrTy, Value *a, Value *b) { + Obj objTy; + Types::PointsToType(ptrTy, &objTy); return B->CreatePtrDiff( #if LLVM_VERSION >= 140 - a->getType()->getPointerElementType(), + getType(&objTy)->type, #endif a, b); } @@ -2219,7 +2221,7 @@ struct FunctionEmitter { Ty->EnsurePointsToCompleteType(&aot); if (bt->type->isPointerTy()) { assert(kind == T_sub); - return emitPointerSub(t, a, b); + return emitPointerSub(&aot, a, b); } else { assert(bt->type->isIntegerTy()); return emitPointerArith(kind, &aot, a, bt, b); From 2a68e2cfea5ab84435a99ed0fca5ea82a68a2675 Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Thu, 24 Aug 2023 16:43:48 -0700 Subject: [PATCH 11/14] Fix another case. --- src/tcompiler.cpp | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/tcompiler.cpp b/src/tcompiler.cpp index 9e421197..9ce35696 100644 --- a/src/tcompiler.cpp +++ b/src/tcompiler.cpp @@ -2388,10 +2388,10 @@ struct FunctionEmitter { // in all cases we simply bitcast cast the resulting pointer to the expected type Obj entryType; entry.obj("type", &entryType); - if (entry.boolean("inunion") || - isPointerToFunction(addr->getType()->getPointerElementType())) { + TType *entryTType = getType(&entryType); + if (entry.boolean("inunion") || isPointerToFunction(entryTType->type)) { unsigned as = addr->getType()->getPointerAddressSpace(); - Type *resultType = PointerType::get(getType(&entryType)->type, as); + Type *resultType = PointerType::get(entryTType->type, as); addr = B->CreateBitCast(addr, resultType); } From 29875bf0436bd9b9794de250051381cf76eeb973 Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Thu, 24 Aug 2023 16:47:12 -0700 Subject: [PATCH 12/14] Fix one more. --- src/tcompiler.cpp | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/src/tcompiler.cpp b/src/tcompiler.cpp index 9ce35696..5e1e92dc 100644 --- a/src/tcompiler.cpp +++ b/src/tcompiler.cpp @@ -2584,16 +2584,15 @@ struct FunctionEmitter { assert(aggType->type->isPointerTy()); Obj objType; Types::PointsToType(&aggTypeO, &objType); + TType *objTType = getType(&objType); idxExp = emitIndex(typeOfValue(&idx), 64, idxExp); // otherwise we have a pointer access which will use a GEP instruction std::vector idxs; Ty->EnsurePointsToCompleteType(&aggTypeO); - Value *result = - B->CreateGEP(getType(&objType)->type, valueExp, idxExp); + Value *result = B->CreateGEP(objTType->type, valueExp, idxExp); if (!exp->boolean("lvalue")) - result = B->CreateLoad(result->getType()->getPointerElementType(), - result); + result = B->CreateLoad(objTType->type, result); return result; } } break; From 2854b0e2b52b7ed1c8291a915ba8536677c86deb Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Thu, 24 Aug 2023 16:53:32 -0700 Subject: [PATCH 13/14] Fix one more. --- src/tcompiler.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/tcompiler.cpp b/src/tcompiler.cpp index 5e1e92dc..268b1c23 100644 --- a/src/tcompiler.cpp +++ b/src/tcompiler.cpp @@ -2616,8 +2616,8 @@ struct FunctionEmitter { } Obj objType; - type.obj("type", &objType); - if (t->type->getPointerElementType()->isIntegerTy(8)) { + Types::PointsToType(&type, &objType); + if (getType(&objType)->type->isIntegerTy(8)) { Obj stringvalue; exp->obj("value", &stringvalue); Value *str = lookupSymbol(CU->symbols, &stringvalue); From 6bba97aa0207bea9135cd83fc7064f88104b3269 Mon Sep 17 00:00:00 2001 From: Elliott Slaughter Date: Thu, 24 Aug 2023 16:58:34 -0700 Subject: [PATCH 14/14] Fix one more. --- src/tcompiler.cpp | 17 +++++++++-------- 1 file changed, 9 insertions(+), 8 deletions(-) diff --git a/src/tcompiler.cpp b/src/tcompiler.cpp index 268b1c23..763d2d98 100644 --- a/src/tcompiler.cpp +++ b/src/tcompiler.cpp @@ -2362,7 +2362,8 @@ struct FunctionEmitter { bool isPointerToFunction(Type *t) { return t->isPointerTy() && t->getPointerElementType()->isFunctionTy(); } - Value *emitStructSelect(Obj *structType, Value *structPtr, int index) { + Value *emitStructSelect(Obj *structType, Value *structPtr, int index, + Obj *entryType) { assert(structPtr->getType()->isPointerTy()); assert(structPtr->getType()->getPointerElementType()->isStructTy()); Ty->EnsureTypeIsComplete(structType); @@ -2386,9 +2387,8 @@ struct FunctionEmitter { // (Terra internal represents functions with i8* for simplicity) // 3. if the field was an anonymous C struct, so we don't know its name // in all cases we simply bitcast cast the resulting pointer to the expected type - Obj entryType; - entry.obj("type", &entryType); - TType *entryTType = getType(&entryType); + entry.obj("type", entryType); + TType *entryTType = getType(entryType); if (entry.boolean("inunion") || isPointerToFunction(entryTType->type)) { unsigned as = addr->getType()->getPointerAddressSpace(); Type *resultType = PointerType::get(entryTType->type, as); @@ -2698,7 +2698,8 @@ struct FunctionEmitter { Obj value; entry.obj("value", &value); int idx = entry.number("index"); - Value *oe = emitStructSelect(&to, output, idx); + Obj entryType; + Value *oe = emitStructSelect(&to, output, idx, &entryType); Value *in = emitExp( &value); // these expressions will select from the // structvariable and perform any casts necessary @@ -2753,11 +2754,11 @@ struct FunctionEmitter { int offset = exp->number("index"); Value *v = emitAddressOf(&obj); - Value *result = emitStructSelect(&typ, v, offset); + Obj entryType; + Value *result = emitStructSelect(&typ, v, offset, &entryType); Type *ttype = getType(&typ)->type; if (!exp->boolean("lvalue")) - result = B->CreateLoad(result->getType()->getPointerElementType(), - result); + result = B->CreateLoad(getType(&entryType)->type, result); return result; } break; case T_constructor: