Skip to content

Commit 26e3cc8

Browse files
committed
Never emit long array literals as inline constant.
Also cleaned up rest of the code a bit. The LLVM 3.1 problem was already fixed before by not doing anything on zero-element arrays anyway.
1 parent 0cdb74a commit 26e3cc8

File tree

1 file changed

+19
-34
lines changed

1 file changed

+19
-34
lines changed

gen/arrays.cpp

+19-34
Original file line numberDiff line numberDiff line change
@@ -445,61 +445,46 @@ llvm::Constant* arrayLiteralToConst(IRState* p, ArrayLiteralExp* ale)
445445

446446
void initializeArrayLiteral(IRState* p, ArrayLiteralExp* ale, LLValue* dstMem)
447447
{
448+
size_t elemCount = ale->elements->dim;
449+
448450
// Don't try to write nothing to a zero-element array, we might represent it
449451
// as a null pointer.
450-
if (ale->elements->dim == 0) return;
452+
if (elemCount == 0) return;
451453

452454
if (isConstLiteral(ale))
453455
{
454-
// allocate room for initializers
455456
llvm::Constant* constarr = arrayLiteralToConst(p, ale);
456457

457-
#if LDC_LLVM_VER == 301
458-
// Simply storing the constant array triggers a problem in LLVM 3.1.
459-
// With -O3 the statement
460-
// void[0] sa0 = (void[0]).init;
461-
// is compiled to
462-
// tail call void @llvm.trap()
463-
// which is not what we want!
464-
// Therefore a global variable is always used.
465-
LLGlobalVariable *gvar = new LLGlobalVariable(
466-
*gIR->module,
467-
constarr->getType(),
468-
true,
469-
LLGlobalValue::InternalLinkage,
470-
constarr,
471-
".constarrayliteral_init"
472-
);
473-
DtoMemCpy(dstMem, gvar, DtoConstSize_t(getTypePaddedSize(constarr->getType())));
474-
#else
475-
// If the type pointed to by dstMem is different from the array type
476-
// then we must assign the value to a global variable.
477-
if (constarr->getType() != dstMem->getType()->getPointerElementType())
458+
// Emit a global for longer arrays, as an inline constant is always
459+
// lowered to a series of movs or similar at the asm level. The
460+
// optimizer can still decide to promote the memcpy intrinsic, so
461+
// the cutoff merely affects compilation speed.
462+
if (elemCount <= 4)
463+
{
464+
DtoStore(constarr, DtoBitCast(dstMem, getPtrToType(constarr->getType())));
465+
}
466+
else
478467
{
479-
LLGlobalVariable *gvar = new LLGlobalVariable(
468+
llvm::GlobalVariable* gvar = new llvm::GlobalVariable(
480469
*gIR->module,
481470
constarr->getType(),
482471
true,
483472
LLGlobalValue::InternalLinkage,
484473
constarr,
485-
".constarrayliteral_init"
474+
".arrayliteral"
486475
);
487476
DtoMemCpy(dstMem, gvar, DtoConstSize_t(getTypePaddedSize(constarr->getType())));
488477
}
489-
else
490-
DtoStore(constarr, dstMem);
491-
#endif
492478
}
493479
else
494480
{
495-
// store elements
496-
for (size_t i = 0; i < ale->elements->dim; ++i)
481+
// Store the elements one by one.
482+
for (size_t i = 0; i < elemCount; ++i)
497483
{
498-
LLValue *elemAddr = DtoGEPi(dstMem, 0, i, "tmp", p->scopebb());
484+
DValue* e = (*ale->elements)[i]->toElem(p);
499485

500-
// emulate assignment
501-
DVarValue *vv = new DVarValue((*ale->elements)[i]->type, elemAddr);
502-
DValue *e = (*ale->elements)[i]->toElem(p);
486+
LLValue* elemAddr = DtoGEPi(dstMem, 0, i, "tmp", p->scopebb());
487+
DVarValue* vv = new DVarValue(e->type, elemAddr);
503488
DtoAssign(ale->loc, vv, e);
504489
}
505490
}

0 commit comments

Comments
 (0)