@@ -445,61 +445,46 @@ llvm::Constant* arrayLiteralToConst(IRState* p, ArrayLiteralExp* ale)
445
445
446
446
void initializeArrayLiteral (IRState* p, ArrayLiteralExp* ale, LLValue* dstMem)
447
447
{
448
+ size_t elemCount = ale->elements ->dim ;
449
+
448
450
// Don't try to write nothing to a zero-element array, we might represent it
449
451
// as a null pointer.
450
- if (ale-> elements -> dim == 0 ) return ;
452
+ if (elemCount == 0 ) return ;
451
453
452
454
if (isConstLiteral (ale))
453
455
{
454
- // allocate room for initializers
455
456
llvm::Constant* constarr = arrayLiteralToConst (p, ale);
456
457
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
478
467
{
479
- LLGlobalVariable * gvar = new LLGlobalVariable (
468
+ llvm::GlobalVariable* gvar = new llvm::GlobalVariable (
480
469
*gIR ->module ,
481
470
constarr->getType (),
482
471
true ,
483
472
LLGlobalValue::InternalLinkage,
484
473
constarr,
485
- " .constarrayliteral_init "
474
+ " .arrayliteral "
486
475
);
487
476
DtoMemCpy (dstMem, gvar, DtoConstSize_t (getTypePaddedSize (constarr->getType ())));
488
477
}
489
- else
490
- DtoStore (constarr, dstMem);
491
- #endif
492
478
}
493
479
else
494
480
{
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)
497
483
{
498
- LLValue *elemAddr = DtoGEPi (dstMem, 0 , i, " tmp " , p-> scopebb () );
484
+ DValue* e = (*ale-> elements )[i]-> toElem (p );
499
485
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);
503
488
DtoAssign (ale->loc , vv, e);
504
489
}
505
490
}
0 commit comments