mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-01 00:33:09 +00:00
Fix bug where array malloc with unexpected computation of the size argument resulted in MallocHelper
identifying the malloc as a non-array malloc. This broke GlobalOpt's optimization of stores of mallocs to global variables. The fix is to classify malloc's into 3 categories: 1. non-array mallocs 2. array mallocs whose array size can be determined 3. mallocs that cannot be determined to be of type 1 or 2 and cannot be optimized getMallocArraySize() returns NULL for category 3, and all users of this function must avoid their malloc optimization if this function returns NULL. Eventually, currently unexpected codegen for computing the malloc's size argument will be supported in isArrayMalloc() and getMallocArraySize(), extending malloc optimizations to those examples. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@84199 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
5814fefc7f
commit
2491ce0353
@ -46,9 +46,9 @@ CallInst* extractMallocCallFromBitCast(Value* I);
|
|||||||
/// matches the malloc call IR generated by CallInst::CreateMalloc(). This
|
/// matches the malloc call IR generated by CallInst::CreateMalloc(). This
|
||||||
/// means that it is a malloc call with one bitcast use AND the malloc call's
|
/// means that it is a malloc call with one bitcast use AND the malloc call's
|
||||||
/// size argument is:
|
/// size argument is:
|
||||||
/// 1. a constant not equal to the malloc's allocated type
|
/// 1. a constant not equal to the size of the malloced type
|
||||||
/// or
|
/// or
|
||||||
/// 2. the result of a multiplication by the malloc's allocated type
|
/// 2. the result of a multiplication by the size of the malloced type
|
||||||
/// Otherwise it returns NULL.
|
/// Otherwise it returns NULL.
|
||||||
/// The unique bitcast is needed to determine the type/size of the array
|
/// The unique bitcast is needed to determine the type/size of the array
|
||||||
/// allocation.
|
/// allocation.
|
||||||
@ -66,18 +66,17 @@ const PointerType* getMallocType(const CallInst* CI);
|
|||||||
/// unique bitcast use, then return NULL.
|
/// unique bitcast use, then return NULL.
|
||||||
const Type* getMallocAllocatedType(const CallInst* CI);
|
const Type* getMallocAllocatedType(const CallInst* CI);
|
||||||
|
|
||||||
/// getMallocArraySize - Returns the array size of a malloc call. The array
|
/// getMallocArraySize - Returns the array size of a malloc call. For array
|
||||||
/// size is computated in 1 of 3 ways:
|
/// mallocs, the size is computated in 1 of 3 ways:
|
||||||
/// 1. If the element type if of size 1, then array size is the argument to
|
/// 1. If the element type is of size 1, then array size is the argument to
|
||||||
/// malloc.
|
/// malloc.
|
||||||
/// 2. Else if the malloc's argument is a constant, the array size is that
|
/// 2. Else if the malloc's argument is a constant, the array size is that
|
||||||
/// argument divided by the element type's size.
|
/// argument divided by the element type's size.
|
||||||
/// 3. Else the malloc argument must be a multiplication and the array size is
|
/// 3. Else the malloc argument must be a multiplication and the array size is
|
||||||
/// the first operand of the multiplication.
|
/// the first operand of the multiplication.
|
||||||
/// This function returns constant 1 if:
|
/// For non-array mallocs, the computed size is constant 1.
|
||||||
/// 1. The malloc call's allocated type cannot be determined.
|
/// This function returns NULL for all mallocs whose array size cannot be
|
||||||
/// 2. IR wasn't created by a call to CallInst::CreateMalloc() with a non-NULL
|
/// determined.
|
||||||
/// ArraySize.
|
|
||||||
Value* getMallocArraySize(CallInst* CI, LLVMContext &Context,
|
Value* getMallocArraySize(CallInst* CI, LLVMContext &Context,
|
||||||
const TargetData* TD);
|
const TargetData* TD);
|
||||||
|
|
||||||
|
@ -130,9 +130,9 @@ static bool isArrayMallocHelper(const CallInst *CI, LLVMContext &Context,
|
|||||||
/// matches the malloc call IR generated by CallInst::CreateMalloc(). This
|
/// matches the malloc call IR generated by CallInst::CreateMalloc(). This
|
||||||
/// means that it is a malloc call with one bitcast use AND the malloc call's
|
/// means that it is a malloc call with one bitcast use AND the malloc call's
|
||||||
/// size argument is:
|
/// size argument is:
|
||||||
/// 1. a constant not equal to the malloc's allocated type
|
/// 1. a constant not equal to the size of the malloced type
|
||||||
/// or
|
/// or
|
||||||
/// 2. the result of a multiplication by the malloc's allocated type
|
/// 2. the result of a multiplication by the size of the malloced type
|
||||||
/// Otherwise it returns NULL.
|
/// Otherwise it returns NULL.
|
||||||
/// The unique bitcast is needed to determine the type/size of the array
|
/// The unique bitcast is needed to determine the type/size of the array
|
||||||
/// allocation.
|
/// allocation.
|
||||||
@ -183,25 +183,60 @@ const Type* llvm::getMallocAllocatedType(const CallInst* CI) {
|
|||||||
return PT ? PT->getElementType() : NULL;
|
return PT ? PT->getElementType() : NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// isSafeToGetMallocArraySize - Returns true if the array size of a malloc can
|
||||||
|
/// be determined. It can be determined in these 3 cases of malloc codegen:
|
||||||
|
/// 1. non-array malloc: The malloc's size argument is a constant and equals the /// size of the type being malloced.
|
||||||
|
/// 2. array malloc: This is a malloc call with one bitcast use AND the malloc
|
||||||
|
/// call's size argument is a constant multiple of the size of the malloced
|
||||||
|
/// type.
|
||||||
|
/// 3. array malloc: This is a malloc call with one bitcast use AND the malloc
|
||||||
|
/// call's size argument is the result of a multiplication by the size of the
|
||||||
|
/// malloced type.
|
||||||
|
/// Otherwise returns false.
|
||||||
|
static bool isSafeToGetMallocArraySize(const CallInst *CI,
|
||||||
|
LLVMContext &Context,
|
||||||
|
const TargetData* TD) {
|
||||||
|
if (!CI)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Type must be known to determine array size.
|
||||||
|
const Type* T = getMallocAllocatedType(CI);
|
||||||
|
if (!T) return false;
|
||||||
|
|
||||||
|
Value* MallocArg = CI->getOperand(1);
|
||||||
|
Constant *ElementSize = ConstantExpr::getSizeOf(T);
|
||||||
|
ElementSize = ConstantExpr::getTruncOrBitCast(ElementSize,
|
||||||
|
MallocArg->getType());
|
||||||
|
|
||||||
|
// First, check if it is a non-array malloc.
|
||||||
|
if (isa<ConstantExpr>(MallocArg) && (MallocArg == ElementSize))
|
||||||
|
return true;
|
||||||
|
|
||||||
|
// Second, check if it can be determined that this is an array malloc.
|
||||||
|
return isArrayMallocHelper(CI, Context, TD);
|
||||||
|
}
|
||||||
|
|
||||||
/// isConstantOne - Return true only if val is constant int 1.
|
/// isConstantOne - Return true only if val is constant int 1.
|
||||||
static bool isConstantOne(Value *val) {
|
static bool isConstantOne(Value *val) {
|
||||||
return isa<ConstantInt>(val) && cast<ConstantInt>(val)->isOne();
|
return isa<ConstantInt>(val) && cast<ConstantInt>(val)->isOne();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// getMallocArraySize - Returns the array size of a malloc call. The array
|
/// getMallocArraySize - Returns the array size of a malloc call. For array
|
||||||
/// size is computated in 1 of 3 ways:
|
/// mallocs, the size is computated in 1 of 3 ways:
|
||||||
/// 1. If the element type if of size 1, then array size is the argument to
|
/// 1. If the element type is of size 1, then array size is the argument to
|
||||||
/// malloc.
|
/// malloc.
|
||||||
/// 2. Else if the malloc's argument is a constant, the array size is that
|
/// 2. Else if the malloc's argument is a constant, the array size is that
|
||||||
/// argument divided by the element type's size.
|
/// argument divided by the element type's size.
|
||||||
/// 3. Else the malloc argument must be a multiplication and the array size is
|
/// 3. Else the malloc argument must be a multiplication and the array size is
|
||||||
/// the first operand of the multiplication.
|
/// the first operand of the multiplication.
|
||||||
/// This function returns constant 1 if:
|
/// For non-array mallocs, the computed size is constant 1.
|
||||||
/// 1. The malloc call's allocated type cannot be determined.
|
/// This function returns NULL for all mallocs whose array size cannot be
|
||||||
/// 2. IR wasn't created by a call to CallInst::CreateMalloc() with a non-NULL
|
/// determined.
|
||||||
/// ArraySize.
|
|
||||||
Value* llvm::getMallocArraySize(CallInst* CI, LLVMContext &Context,
|
Value* llvm::getMallocArraySize(CallInst* CI, LLVMContext &Context,
|
||||||
const TargetData* TD) {
|
const TargetData* TD) {
|
||||||
|
if (isSafeToGetMallocArraySize(CI, Context, TD))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
// Match CreateMalloc's use of constant 1 array-size for non-array mallocs.
|
// Match CreateMalloc's use of constant 1 array-size for non-array mallocs.
|
||||||
if (!isArrayMalloc(CI, Context, TD))
|
if (!isArrayMalloc(CI, Context, TD))
|
||||||
return ConstantInt::get(CI->getOperand(1)->getType(), 1);
|
return ConstantInt::get(CI->getOperand(1)->getType(), 1);
|
||||||
|
@ -102,8 +102,9 @@ const SCEV *PointerTracking::computeAllocationCount(Value *P,
|
|||||||
|
|
||||||
if (CallInst *CI = extractMallocCall(V)) {
|
if (CallInst *CI = extractMallocCall(V)) {
|
||||||
Value *arraySize = getMallocArraySize(CI, P->getContext(), TD);
|
Value *arraySize = getMallocArraySize(CI, P->getContext(), TD);
|
||||||
Ty = getMallocAllocatedType(CI);
|
const Type* AllocTy = getMallocAllocatedType(CI);
|
||||||
if (!Ty || !arraySize) return SE->getCouldNotCompute();
|
if (!AllocTy || !arraySize) return SE->getCouldNotCompute();
|
||||||
|
Ty = AllocTy;
|
||||||
// arraySize elements of type Ty.
|
// arraySize elements of type Ty.
|
||||||
return SE->getSCEV(arraySize);
|
return SE->getSCEV(arraySize);
|
||||||
}
|
}
|
||||||
|
@ -950,12 +950,14 @@ static GlobalVariable *OptimizeGlobalAddressOfMalloc(GlobalVariable *GV,
|
|||||||
BitCastInst *BCI,
|
BitCastInst *BCI,
|
||||||
LLVMContext &Context,
|
LLVMContext &Context,
|
||||||
TargetData* TD) {
|
TargetData* TD) {
|
||||||
|
DEBUG(errs() << "PROMOTING MALLOC GLOBAL: " << *GV
|
||||||
|
<< " CALL = " << *CI << " BCI = " << *BCI << '\n');
|
||||||
|
|
||||||
const Type *IntPtrTy = TD->getIntPtrType(Context);
|
const Type *IntPtrTy = TD->getIntPtrType(Context);
|
||||||
|
|
||||||
DEBUG(errs() << "PROMOTING MALLOC GLOBAL: " << *GV << " MALLOC = " << *CI);
|
Value* ArraySize = getMallocArraySize(CI, Context, TD);
|
||||||
|
assert(ArraySize && "not a malloc whose array size can be determined");
|
||||||
ConstantInt *NElements = cast<ConstantInt>(getMallocArraySize(CI,
|
ConstantInt *NElements = cast<ConstantInt>(ArraySize);
|
||||||
Context, TD));
|
|
||||||
if (NElements->getZExtValue() != 1) {
|
if (NElements->getZExtValue() != 1) {
|
||||||
// If we have an array allocation, transform it to a single element
|
// If we have an array allocation, transform it to a single element
|
||||||
// allocation to make the code below simpler.
|
// allocation to make the code below simpler.
|
||||||
@ -976,9 +978,6 @@ static GlobalVariable *OptimizeGlobalAddressOfMalloc(GlobalVariable *GV,
|
|||||||
|
|
||||||
// Create the new global variable. The contents of the malloc'd memory is
|
// Create the new global variable. The contents of the malloc'd memory is
|
||||||
// undefined, so initialize with an undef value.
|
// undefined, so initialize with an undef value.
|
||||||
// FIXME: This new global should have the alignment returned by malloc. Code
|
|
||||||
// could depend on malloc returning large alignment (on the mac, 16 bytes) but
|
|
||||||
// this would only guarantee some lower alignment.
|
|
||||||
const Type *MAT = getMallocAllocatedType(CI);
|
const Type *MAT = getMallocAllocatedType(CI);
|
||||||
Constant *Init = UndefValue::get(MAT);
|
Constant *Init = UndefValue::get(MAT);
|
||||||
GlobalVariable *NewGV = new GlobalVariable(*GV->getParent(),
|
GlobalVariable *NewGV = new GlobalVariable(*GV->getParent(),
|
||||||
@ -1892,16 +1891,17 @@ static bool TryToOptimizeStoreOfMallocToGlobal(GlobalVariable *GV,
|
|||||||
// transform the program to use global memory instead of malloc'd memory.
|
// transform the program to use global memory instead of malloc'd memory.
|
||||||
// This eliminates dynamic allocation, avoids an indirection accessing the
|
// This eliminates dynamic allocation, avoids an indirection accessing the
|
||||||
// data, and exposes the resultant global to further GlobalOpt.
|
// data, and exposes the resultant global to further GlobalOpt.
|
||||||
if (ConstantInt *NElements =
|
Value *NElems = getMallocArraySize(CI, Context, TD);
|
||||||
dyn_cast<ConstantInt>(getMallocArraySize(CI, Context, TD))) {
|
if (NElems) {
|
||||||
// Restrict this transformation to only working on small allocations
|
if (ConstantInt *NElements = dyn_cast<ConstantInt>(NElems))
|
||||||
// (2048 bytes currently), as we don't want to introduce a 16M global or
|
// Restrict this transformation to only working on small allocations
|
||||||
// something.
|
// (2048 bytes currently), as we don't want to introduce a 16M global or
|
||||||
if (TD &&
|
// something.
|
||||||
NElements->getZExtValue() * TD->getTypeAllocSize(AllocTy) < 2048) {
|
if (TD &&
|
||||||
GVI = OptimizeGlobalAddressOfMalloc(GV, CI, BCI, Context, TD);
|
NElements->getZExtValue() * TD->getTypeAllocSize(AllocTy) < 2048) {
|
||||||
return true;
|
GVI = OptimizeGlobalAddressOfMalloc(GV, CI, BCI, Context, TD);
|
||||||
}
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the allocation is an array of structures, consider transforming this
|
// If the allocation is an array of structures, consider transforming this
|
||||||
|
Loading…
Reference in New Issue
Block a user