blob: d4c2554f0693042e0182892d00274f8288026289 [file] [log] [blame]
//===----------- JavaJIT.h - Java just in time compiler -------------------===//
//
// JnJVM
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
#ifndef JNJVM_JAVA_JIT_H
#define JNJVM_JAVA_JIT_H
#include <vector>
#include <map>
#include "llvm/BasicBlock.h"
#include "llvm/DerivedTypes.h"
#include "llvm/Function.h"
#include "llvm/Instructions.h"
#include "llvm/Value.h"
#include "types.h"
#include "mvm/Object.h"
#include "mvm/PrintBuffer.h"
#include "JnjvmModule.h"
namespace jnjvm {
class Class;
class JavaMethod;
class JavaObject;
class Jnjvm;
class Reader;
class UTF8;
struct Exception {
uint32 startpc;
uint32 endpc;
uint32 handlerpc;
uint16 catche;
UserClass* catchClass;
llvm::BasicBlock* test;
llvm::BasicBlock* realTest;
llvm::BasicBlock* javaHandler;
llvm::BasicBlock* nativeHandler;
llvm::PHINode* exceptionPHI;
llvm::PHINode* handlerPHI;
};
struct Opinfo {
llvm::BasicBlock* newBlock;
bool reqSuppl;
llvm::BasicBlock* exceptionBlock;
};
class JavaJIT {
private:
llvm::Value* getConstantPoolAt(uint32 index, llvm::Function* resolver,
const llvm::Type* returnType,
llvm::Value* addArg, bool doThrow = true);
public:
JavaJIT(JavaMethod* meth, llvm::Function* func) {
nbEnveloppes = 0;
compilingMethod = meth;
compilingClass = meth->classDef;
module = compilingClass->classLoader->getModule();
llvmFunction = func;
inlining = false;
callsStackWalker = false;
}
JnjvmModule* module;
static void invokeOnceVoid(Jnjvm* vm, JnjvmClassLoader* loader,
const char* className,
const char* func, const char* sign,
int access, ...);
llvm::Function* javaCompile();
llvm::Function* nativeCompile(void* natPtr = 0);
llvm::Instruction* inlineCompile(llvm::BasicBlock*& curBB,
llvm::BasicBlock* endExBlock,
std::vector<llvm::Value*>& args);
std::map<JavaMethod*, bool> inlineMethods;
bool inlining;
Class* compilingClass;
JavaMethod* compilingMethod;
llvm::Function* llvmFunction;
const llvm::Type* returnType;
// change into LLVM instructions
void compileOpcodes(uint8* bytecodes, uint32 codeLength);
// create basic blocks
void exploreOpcodes(uint8* bytecodes, uint32 codeLength);
// load constant
void _ldc(uint16 index);
// float comparisons
void compareFP(llvm::Value*, llvm::Value*, const llvm::Type*, bool l);
// null pointer exception
void JITVerifyNull(llvm::Value* obj);
// stack manipulation
std::vector< std::pair<llvm::Value*, bool> > stack;
void push(llvm::Value* val, bool unsign) {
stack.push_back(std::make_pair(val, unsign));
}
void push(std::pair<llvm::Value*, bool> pair) {
stack.push_back(pair);
}
llvm::Value* pop() {
llvm::Value * ret = top();
stack.pop_back();
return ret;
}
llvm::Value* top() {
return stack.back().first;
}
bool topFunc() {
return stack.back().second;
}
uint32 stackSize() {
return stack.size();
}
llvm::Value* popAsInt() {
llvm::Value * ret = top();
bool unsign = topFunc();
stack.pop_back();
if (ret->getType() != llvm::Type::Int32Ty) {
if (unsign) {
ret = new llvm::ZExtInst(ret, llvm::Type::Int32Ty, "", currentBlock);
} else {
ret = new llvm::SExtInst(ret, llvm::Type::Int32Ty, "", currentBlock);
}
}
return ret;
}
std::pair<llvm::Value*, bool> popPair() {
std::pair<llvm::Value*, bool> ret = stack.back();
stack.pop_back();
return ret;
}
// exceptions
std::vector<llvm::BasicBlock*> jsrs;
// exception local
llvm::Value* supplLocal;
unsigned readExceptionTable(Reader& reader);
llvm::BasicBlock* endExceptionBlock;
llvm::BasicBlock* currentExceptionBlock;
llvm::BasicBlock* unifiedUnreachable;
Opinfo* opcodeInfos;
// block manipulation
llvm::BasicBlock* currentBlock;
llvm::BasicBlock* createBasicBlock(const char* name = "") {
return llvm::BasicBlock::Create(name, llvmFunction);
}
void setCurrentBlock(llvm::BasicBlock* block);
// branches
void branch(llvm::Value* test, llvm::BasicBlock* ifTrue,
llvm::BasicBlock* ifFalse, llvm::BasicBlock* insert);
void branch(llvm::BasicBlock* where, llvm::BasicBlock* insert);
// locals
std::vector<llvm::Value*> intLocals;
std::vector<llvm::Value*> longLocals;
std::vector<llvm::Value*> floatLocals;
std::vector<llvm::Value*> doubleLocals;
std::vector<llvm::Value*> objectLocals;
// end function
llvm::BasicBlock* endBlock;
llvm::PHINode* endNode;
// array manipulation
llvm::Value* verifyAndComputePtr(llvm::Value* obj, llvm::Value* index,
const llvm::Type* arrayType,
bool verif = true);
llvm::Value* arraySize(llvm::Value* obj);
// synchronize
void beginSynchronize();
void endSynchronize();
void monitorEnter(llvm::Value* obj);
void monitorExit(llvm::Value* obj);
// fields invoke
void getStaticField(uint16 index);
void setStaticField(uint16 index);
void getVirtualField(uint16 index);
void setVirtualField(uint16 index);
llvm::Value* ldResolved(uint16 index, bool stat, llvm::Value* object,
const llvm::Type* fieldType,
const llvm::Type* fieldTypePtr);
llvm::Value* getResolvedClass(uint16 index, bool clinit, bool doThrow = true);
// methods invoke
void makeArgs(llvm::FunctionType::param_iterator it,
uint32 index, std::vector<llvm::Value*>& result, uint32 nb);
void invokeVirtual(uint16 index);
void invokeInterfaceOrVirtual(uint16 index);
void invokeSpecial(uint16 index);
void invokeStatic(uint16 index);
void invokeNew(uint16 index);
llvm::Instruction* invokeInline(JavaMethod* meth,
std::vector<llvm::Value*>& args);
llvm::Instruction* lowerMathOps(const UTF8* name,
std::vector<llvm::Value*>& args);
llvm::Instruction* invoke(llvm::Value *F, std::vector<llvm::Value*>&args,
const char* Name,
llvm::BasicBlock *InsertAtEnd);
// Alternate CallInst ctors w/ two actuals, w/ one actual and no
// actuals, respectively.
llvm::Instruction* invoke(llvm::Value *F, llvm::Value *Actual1,
llvm::Value *Actual2, const char* Name,
llvm::BasicBlock *InsertAtEnd);
llvm::Instruction* invoke(llvm::Value *F, llvm::Value *Actual1,
const char* Name, llvm::BasicBlock *InsertAtEnd);
llvm::Instruction* invoke(llvm::Value *F, const char* Name,
llvm::BasicBlock *InsertAtEnd);
void convertValue(llvm::Value*& val, const llvm::Type* t1,
llvm::BasicBlock* currentBlock, bool usign);
// wide status
bool wide;
uint32 WREAD_U1(uint8* bytecodes, bool init, uint32 &i);
sint32 WREAD_S1(uint8* bytecodes, bool init, uint32 &i);
uint32 WCALC(uint32 n);
uint16 maxStack;
uint16 maxLocals;
uint32 codeLen;
#if defined(ISOLATE) || defined(MULTIPLE_GC)
llvm::Value* isolateLocal;
#if defined(ISOLATE_SHARING)
llvm::Value* ctpCache;
llvm::Value* getStaticInstanceCtp();
llvm::Value* getClassCtp();
#endif
#endif
static const char* OpcodeNames[256];
static UserClass* getCallingClass();
static UserClass* getCallingClassWalker();
static JavaObject* getCallingClassLoader();
static void printBacktrace();
static JavaMethod* IPToJavaMethod(void* ip);
/// nbEnveloppes - Number of enveloppes (ie invokeinterface) in this
/// method.
uint32 nbEnveloppes;
bool canBeInlined(JavaMethod* meth);
bool callsStackWalker;
};
enum Opcode {
NOP = 0x00,
ACONST_NULL = 0x01,
ICONST_M1 = 0x02,
ICONST_0 = 0x03,
ICONST_1 = 0x04,
ICONST_2 = 0x05,
ICONST_3 = 0x06,
ICONST_4 = 0x07,
ICONST_5 = 0x08,
LCONST_0 = 0x09,
LCONST_1 = 0x0A,
FCONST_0 = 0x0B,
FCONST_1 = 0x0C,
FCONST_2 = 0x0D,
DCONST_0 = 0x0E,
DCONST_1 = 0x0F,
BIPUSH = 0x10,
SIPUSH = 0x11,
LDC = 0x12,
LDC_W = 0x13,
LDC2_W = 0x14,
ILOAD = 0x15,
LLOAD = 0x16,
FLOAD = 0x17,
DLOAD = 0x18,
ALOAD = 0x19,
ILOAD_0 = 0x1A,
ILOAD_1 = 0x1B,
ILOAD_2 = 0x1C,
ILOAD_3 = 0x1D,
LLOAD_0 = 0x1E,
LLOAD_1 = 0x1F,
LLOAD_2 = 0x20,
LLOAD_3 = 0x21,
FLOAD_0 = 0x22,
FLOAD_1 = 0x23,
FLOAD_2 = 0x24,
FLOAD_3 = 0x25,
DLOAD_0 = 0x26,
DLOAD_1 = 0x27,
DLOAD_2 = 0x28,
DLOAD_3 = 0x29,
ALOAD_0 = 0x2A,
ALOAD_1 = 0x2B,
ALOAD_2 = 0x2C,
ALOAD_3 = 0x2D,
IALOAD = 0x2E,
LALOAD = 0x2F,
FALOAD = 0x30,
DALOAD = 0x31,
AALOAD = 0x32,
BALOAD = 0x33,
CALOAD = 0x34,
SALOAD = 0x35,
ISTORE = 0x36,
LSTORE = 0x37,
FSTORE = 0x38,
DSTORE = 0x39,
ASTORE = 0x3A,
ISTORE_0 = 0x3B,
ISTORE_1 = 0x3C,
ISTORE_2 = 0x3D,
ISTORE_3 = 0x3E,
LSTORE_0 = 0x3F,
LSTORE_1 = 0x40,
LSTORE_2 = 0x41,
LSTORE_3 = 0x42,
FSTORE_0 = 0x43,
FSTORE_1 = 0x44,
FSTORE_2 = 0x45,
FSTORE_3 = 0x46,
DSTORE_0 = 0x47,
DSTORE_1 = 0x48,
DSTORE_2 = 0x49,
DSTORE_3 = 0x4A,
ASTORE_0 = 0x4B,
ASTORE_1 = 0x4C,
ASTORE_2 = 0x4D,
ASTORE_3 = 0x4E,
IASTORE = 0x4F,
LASTORE = 0x50,
FASTORE = 0x51,
DASTORE = 0x52,
AASTORE = 0x53,
BASTORE = 0x54,
CASTORE = 0x55,
SASTORE = 0x56,
POP = 0x57,
POP2 = 0x58,
DUP = 0x59,
DUP_X1 = 0x5A,
DUP_X2 = 0x5B,
DUP2 = 0x5C,
DUP2_X1 = 0x5D,
DUP2_X2 = 0x5E,
SWAP = 0x5F,
IADD = 0x60,
LADD = 0x61,
FADD = 0x62,
DADD = 0x63,
ISUB = 0x64,
LSUB = 0x65,
FSUB = 0x66,
DSUB = 0x67,
IMUL = 0x68,
LMUL = 0x69,
FMUL = 0x6A,
DMUL = 0x6B,
IDIV = 0x6C,
LDIV = 0x6D,
FDIV = 0x6E,
DDIV = 0x6F,
IREM = 0x70,
LREM = 0x71,
FREM = 0x72,
DREM = 0x73,
INEG = 0x74,
LNEG = 0x75,
FNEG = 0x76,
DNEG = 0x77,
ISHL = 0x78,
LSHL = 0x79,
ISHR = 0x7A,
LSHR = 0x7B,
IUSHR = 0x7C,
LUSHR = 0x7D,
IAND = 0x7E,
LAND = 0x7F,
IOR = 0x80,
LOR = 0x81,
IXOR = 0x82,
LXOR = 0x83,
IINC = 0x84,
I2L = 0x85,
I2F = 0x86,
I2D = 0x87,
L2I = 0x88,
L2F = 0x89,
L2D = 0x8A,
F2I = 0x8B,
F2L = 0x8C,
F2D = 0x8D,
D2I = 0x8E,
D2L = 0x8F,
D2F = 0x90,
I2B = 0x91,
I2C = 0x92,
I2S = 0x93,
LCMP = 0x94,
FCMPL = 0x95,
FCMPG = 0x96,
DCMPL = 0x97,
DCMPG = 0x98,
IFEQ = 0x99,
IFNE = 0x9A,
IFLT = 0x9B,
IFGE = 0x9C,
IFGT = 0x9D,
IFLE = 0x9E,
IF_ICMPEQ = 0x9F,
IF_ICMPNE = 0xA0,
IF_ICMPLT = 0xA1,
IF_ICMPGE = 0xA2,
IF_ICMPGT = 0xA3,
IF_ICMPLE = 0xA4,
IF_ACMPEQ = 0xA5,
IF_ACMPNE = 0xA6,
GOTO = 0xA7,
JSR = 0xA8,
RET = 0xA9,
TABLESWITCH = 0xAA,
LOOKUPSWITCH = 0xAB,
IRETURN = 0xAC,
LRETURN = 0xAD,
FRETURN = 0xAE,
DRETURN = 0xAF,
ARETURN = 0xB0,
RETURN = 0xB1,
GETSTATIC = 0xB2,
PUTSTATIC = 0xB3,
GETFIELD = 0xB4,
PUTFIELD = 0xB5,
INVOKEVIRTUAL = 0xB6,
INVOKESPECIAL = 0xB7,
INVOKESTATIC = 0xB8,
INVOKEINTERFACE = 0xB9,
UNUSED = 0xBA,
NEW = 0xBB,
NEWARRAY = 0xBC,
ANEWARRAY = 0xBD,
ARRAYLENGTH = 0xBE,
ATHROW = 0xBF,
CHECKCAST = 0xC0,
INSTANCEOF = 0xC1,
MONITORENTER = 0xC2,
MONITOREXIT = 0xC3,
WIDE = 0xC4,
MULTIANEWARRAY = 0xC5,
IFNULL = 0xC6,
IFNONNULL = 0xC7,
GOTO_W = 0xC8,
JSR_W = 0xC9,
BREAKPOINT = 0xCA,
IMPDEP1 = 0xFE,
IMPDEP2 = 0xFF
};
} // end namespace jnjvm
#endif