/*
* Copyright (C) 2012 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef ART_COMPILER_LLVM_INTRINSIC_HELPER_H_
#define ART_COMPILER_LLVM_INTRINSIC_HELPER_H_
#include "base/logging.h"
#include <llvm/ADT/DenseMap.h>
namespace llvm {
class Function;
class FunctionType;
class LLVMContext;
class Module;
} // namespace llvm
namespace art {
namespace llvm {
class IRBuilder;
class IntrinsicHelper {
public:
enum IntrinsicId {
#define DEF_INTRINSICS_FUNC(ID, ...) ID,
#include "intrinsic_func_list.def"
MaxIntrinsicId,
// Pseudo-intrinsics Id
UnknownId
};
enum IntrinsicAttribute {
kAttrNone = 0,
// Intrinsic that neither modified the memory state nor refer to the global
// state
kAttrReadNone = 1 << 0,
// Intrinsic that doesn't modify the memory state. Note that one should set
// this flag carefully when the intrinsic may throw exception. Since the
// thread state is implicitly modified when an exception is thrown.
kAttrReadOnly = 1 << 1,
// Note that intrinsic without kAttrNoThrow and kAttrDoThrow set means that
// intrinsic generates exception in some cases
// Intrinsic that never generates exception
kAttrNoThrow = 1 << 2,
// Intrinsic that always generate exception
kAttrDoThrow = 1 << 3,
};
enum IntrinsicValType {
kNone,
kVoidTy,
kJavaObjectTy,
kJavaMethodTy,
kJavaThreadTy,
kInt1Ty,
kInt8Ty,
kInt16Ty,
kInt32Ty,
kInt64Ty,
kFloatTy,
kDoubleTy,
kInt1ConstantTy,
kInt8ConstantTy,
kInt16ConstantTy,
kInt32ConstantTy,
kInt64ConstantTy,
kFloatConstantTy,
kDoubleConstantTy,
kVarArgTy,
};
enum {
kIntrinsicMaxArgc = 5
};
typedef struct IntrinsicInfo {
const char* name_;
unsigned attr_;
IntrinsicValType ret_val_type_;
IntrinsicValType arg_type_[kIntrinsicMaxArgc];
} IntrinsicInfo;
private:
static const IntrinsicInfo Info[];
public:
static const IntrinsicInfo& GetInfo(IntrinsicId id) {
DCHECK(id >= 0 && id < MaxIntrinsicId) << "Unknown ART intrinsics ID: "
<< id;
return Info[id];
}
static const char* GetName(IntrinsicId id) {
return (id <= MaxIntrinsicId) ? GetInfo(id).name_ : "InvalidIntrinsic";
}
static unsigned GetAttr(IntrinsicId id) {
return GetInfo(id).attr_;
}
public:
IntrinsicHelper(::llvm::LLVMContext& context, ::llvm::Module& module);
::llvm::Function* GetIntrinsicFunction(IntrinsicId id) {
DCHECK(id >= 0 && id < MaxIntrinsicId) << "Unknown ART intrinsics ID: "
<< id;
return intrinsic_funcs_[id];
}
IntrinsicId GetIntrinsicId(const ::llvm::Function* func) const {
::llvm::DenseMap<const ::llvm::Function*, IntrinsicId>::const_iterator
i = intrinsic_funcs_map_.find(func);
if (i == intrinsic_funcs_map_.end()) {
return UnknownId;
} else {
return i->second;
}
}
private:
// FIXME: "+1" is to workaround the GCC bugs:
// http://gcc.gnu.org/bugzilla/show_bug.cgi?id=43949
// Remove this when uses newer GCC (> 4.4.3)
::llvm::Function* intrinsic_funcs_[MaxIntrinsicId + 1];
// Map a llvm::Function to its intrinsic id
::llvm::DenseMap<const ::llvm::Function*, IntrinsicId> intrinsic_funcs_map_;
};
} // namespace llvm
} // namespace art
#endif // ART_COMPILER_LLVM_INTRINSIC_HELPER_H_