blob: 8076669f45ae590516535215c015e3d87fa9900c [file] [log] [blame]
//===--- Util.cpp ---------------------------------------------------------===//
//
// Copyright 2018 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
//
//===----------------------------------------------------------------------===//
//
// Helper functions.
//
//===----------------------------------------------------------------------===//
#include "GollvmPasses.h"
#include "llvm/ADT/ArrayRef.h"
#include "llvm/ADT/BitVector.h"
#include "llvm/ADT/SmallVector.h"
#include "llvm/IR/Constants.h"
#include "llvm/IR/DataLayout.h"
#include "llvm/IR/Type.h"
#include "llvm/IR/Value.h"
using namespace llvm;
using namespace gollvm::passes;
// Whether a type contains pointer.
bool
gollvm::passes::hasPointer(Type *T) {
switch (T->getTypeID()) {
case Type::PointerTyID:
return true;
case Type::ArrayTyID:
return hasPointer(T->getArrayElementType());
case Type::VectorTyID:
return hasPointer(T->getVectorElementType());
case Type::StructTyID: {
for (unsigned i = 0, e = T->getStructNumElements(); i < e; ++i)
if (hasPointer(T->getStructElementType(i)))
return true;
return false;
}
default:
return false;
}
}
static void
getPtrBitmapForTypeHelper(Type *T, const DataLayout &DL, uint64_t BaseOffset, BitVector &BV) {
if (!hasPointer(T))
return;
const unsigned PtrSize = DL.getPointerSize();
Type *Int32Ty = Type::getInt32Ty(T->getContext());
switch (T->getTypeID()) {
case Type::PointerTyID:
BV.set(BaseOffset / PtrSize);
break;;
case Type::ArrayTyID: {
Type *ET = T->getArrayElementType();
for (unsigned i = 0, n = T->getArrayNumElements(); i < n; ++i) {
ArrayRef<Value*> Idx = { ConstantInt::get(Int32Ty, 0), ConstantInt::get(Int32Ty, i) };
uint64_t Offset = DL.getIndexedOffsetInType(T, Idx);
getPtrBitmapForTypeHelper(ET, DL, BaseOffset+Offset, BV);
}
break;
}
case Type::VectorTyID: {
Type *ET = T->getVectorElementType();
for (unsigned i = 0, n = T->getVectorNumElements(); i < n; ++i) {
ArrayRef<Value*> Idx = { ConstantInt::get(Int32Ty, 0), ConstantInt::get(Int32Ty, i) };
uint64_t Offset = DL.getIndexedOffsetInType(T, Idx);
getPtrBitmapForTypeHelper(ET, DL, BaseOffset+Offset, BV);
}
break;
}
case Type::StructTyID: {
for (unsigned i = 0, n = T->getStructNumElements(); i < n; ++i) {
Type *ET = T->getStructElementType(i);
if (!hasPointer(ET))
continue;
ArrayRef<Value*> Idx = { ConstantInt::get(Int32Ty, 0), ConstantInt::get(Int32Ty, i) };
uint64_t Offset = DL.getIndexedOffsetInType(T, Idx);
getPtrBitmapForTypeHelper(ET, DL, BaseOffset+Offset, BV);
}
break;
}
default:
break;
}
}
// Compute the pointer bitmap for type T, stored into Words.
void
gollvm::passes::getPtrBitmapForType(Type *T, const DataLayout &DL,
SmallVectorImpl<Value *> &Words) {
// TODO: this function is silly -- BitVector internally has
// a bitmap storage, but it is private. Can we do better?
const unsigned PtrSize = DL.getPointerSize();
Type *Int32Ty = Type::getInt32Ty(T->getContext());
uint64_t Size = DL.getTypeStoreSize(T);
BitVector BV(Size/PtrSize);
getPtrBitmapForTypeHelper(T, DL, 0, BV);
if (BV.none())
return;
unsigned last = BV.find_last();
if (last == 0) // a single pointer field, no need of a bitmap
return;
//Words.reserve(last/32 + 1);
for (unsigned i = 0; i <= last; i += 32) {
uint32_t w = 0;
for (unsigned j = 0; j < 32 && i+j <= last; j++)
w |= BV[i+j] ? 1<<j : 0;
Words.push_back(ConstantInt::get(Int32Ty, w));
}
}