⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 interp-direct.h

📁 jam vm 最新Java虚拟机!为开源项目源代码
💻 H
字号:
/* * Copyright (C) 2003, 2004, 2005, 2006, 2007 * Robert Lougher <rob@lougher.org.uk>. * * This file is part of JamVM. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2, * or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */#ifndef THREADED#error Direct interpreter cannot be built non-threaded#endif/* Macros for handler/bytecode rewriting */#ifdef USE_CACHE#define WITH_OPCODE_CHANGE_CP_DINDEX(opcode, index, cache) \{                                                          \    index = pc->operand.uui.u1;                            \    cache = pc->operand.uui.i;                             \    if(pc->handler != L(opcode, 0, ENTRY) &&               \       pc->handler != L(opcode, 1, ENTRY) &&               \       pc->handler != L(opcode, 2, ENTRY))                 \        REDISPATCH                                         \}#else /* USE_CACHE */#define WITH_OPCODE_CHANGE_CP_DINDEX(opcode, index, cache) \{                                                          \    index = pc->operand.uui.u1;                            \    cache = pc->operand.uui.i;                             \    if(pc->handler != L(opcode, 0, ENTRY))                 \        REDISPATCH                                         \}#endif#define OPCODE_REWRITE(opcode, cache, new_operand)         \{                                                          \    pc->handler = &&rewrite_lock;                          \    MBARRIER();                                            \    pc->operand = new_operand;                             \    MBARRIER();                                            \    pc->handler = handlers[cache][opcode];                 \}/* Two levels of macros are needed to correctly produce the label * from the OPC_xxx macro passed into DEF_OPC as cpp doesn't  * prescan when concatenating with ##... * * On gcc <= 2.95, we also get a space inserted before the : * e.g DEF_OPC(OPC_NULL) -> opc0 : - the ##: is a hack to fix * this, but this generates warnings on >= 2.96... */#if (__GNUC__ == 2) && (__GNUC_MINOR__ <= 95)#define label(x, y, z)                          \opc##x##_##y##_##z##:#else#define label(x, y, z)                          \opc##x##_##y##_##z:#endif#define DEF_OPC(opcode, level, BODY)            \    label(opcode, level, ENTRY)                 \        BODY#define DEF_OPC_2(op1, op2, level, BODY)        \    label(op1, level, ENTRY)                    \    label(op2, level, ENTRY)                    \        BODY#define DEF_OPC_3(op1, op2, op3, level, BODY)   \    label(op1, level, ENTRY)                    \    label(op2, level, ENTRY)                    \    label(op3, level, ENTRY)                    \        BODY#ifdef USE_CACHE#define DEF_OPC_012(opcode, BODY)               \    label(opcode, 0, ENTRY)                     \        cache.i.v1 = *--ostack;                 \                                                \    label(opcode, 1, ENTRY)                     \        cache.i.v2 = cache.i.v1;                \        cache.i.v1 = *--ostack;                 \                                                \    label(opcode, 2, ENTRY)                     \        BODY        #define DEF_OPC_012_2(op1, op2, BODY)           \    label(op1, 0, ENTRY)                        \    label(op2, 0, ENTRY)                        \        cache.i.v1 = *--ostack;                 \                                                \    label(op1, 1, ENTRY)                        \    label(op2, 1, ENTRY)                        \        cache.i.v2 = cache.i.v1;                \        cache.i.v1 = *--ostack;                 \                                                \    label(op1, 2, ENTRY)                        \    label(op2, 2, ENTRY)                        \        BODY#define DEF_OPC_012_3(op1, op2, op3, BODY)      \    label(op1, 0, ENTRY)                        \    label(op2, 0, ENTRY)                        \    label(op3, 0, ENTRY)                        \        cache.i.v1 = *--ostack;                 \                                                \    label(op1, 1, ENTRY)                        \    label(op2, 1, ENTRY)                        \    label(op3, 1, ENTRY)                        \        cache.i.v2 = cache.i.v1;                \        cache.i.v1 = *--ostack;                 \                                                \    label(op1, 2, ENTRY)                        \    label(op2, 2, ENTRY)                        \    label(op3, 2, ENTRY)                        \        BODY#define DEF_OPC_012_4(op1, op2, op3, op4, BODY) \    label(op1, 0, ENTRY)                        \    label(op2, 0, ENTRY)                        \    label(op3, 0, ENTRY)                        \    label(op4, 0, ENTRY)                        \        cache.i.v1 = *--ostack;                 \                                                \    label(op1, 1, ENTRY)                        \    label(op2, 1, ENTRY)                        \    label(op3, 1, ENTRY)                        \    label(op4, 1, ENTRY)                        \        cache.i.v2 = cache.i.v1;                \        cache.i.v1 = *--ostack;                 \                                                \    label(op1, 2, ENTRY)                        \    label(op2, 2, ENTRY)                        \    label(op3, 2, ENTRY)                        \    label(op4, 2, ENTRY)                        \        BODY#define DEF_OPC_210(opcode, BODY)               \    label(opcode, 2, ENTRY)                     \        *ostack++ = cache.i.v1;                 \        cache.i.v1 = cache.i.v2;                \                                                \    label(opcode, 1, ENTRY)                     \        *ostack++ = cache.i.v1;                 \                                                \    label(opcode, 0, ENTRY)                     \        BODY        #define DEF_OPC_210_2(op1, op2, BODY)           \    label(op1, 2, ENTRY)                        \    label(op2, 2, ENTRY)                        \        *ostack++ = cache.i.v1;                 \        cache.i.v1 = cache.i.v2;                \                                                \    label(op1, 1, ENTRY)                        \    label(op2, 1, ENTRY)                        \        *ostack++ = cache.i.v1;                 \                                                \    label(op1, 0, ENTRY)                        \    label(op2, 0, ENTRY)                        \        BODY#define LABELS(opcode)                          \    label(opcode, 0, ENTRY)                     \    label(opcode, 1, ENTRY)                     \    label(opcode, 2, ENTRY)#define DEF_OPC_RW(opcode, BODY)                \    LABELS(opcode)                              \        BODY#define DEF_OPC_RW_4(op1, op2, op3, op4, BODY)  \    LABELS(op1)                                 \    LABELS(op2)                                 \    LABELS(op3)                                 \    LABELS(op4)                                 \        BODY#else /* USE_CACHE */#define DEF_OPC_012(opcode, BODY)               \    DEF_OPC(opcode, 0, BODY)#define DEF_OPC_012_2(op1, op2, BODY)           \    DEF_OPC_2(op1, op2, 0, BODY)#define DEF_OPC_012_3(op1, op2, op3, BODY)      \    DEF_OPC_3(op1, op2, op3, 0, BODY)#define DEF_OPC_012_4(op1, op2, op3, op4, BODY) \    label(op1, 0, ENTRY)                        \    label(op2, 0, ENTRY)                        \    label(op3, 0, ENTRY)                        \    label(op4, 0, ENTRY)                        \        BODY#define DEF_OPC_210(opcode, BODY)               \    DEF_OPC_012(opcode, ({BODY});)#define DEF_OPC_210_2(op1, op2, BODY)           \    DEF_OPC_012_2(op1, op2, ({BODY});)#define DEF_OPC_RW(opcode, BODY)                \    DEF_OPC_012(opcode, ({BODY});)#define DEF_OPC_RW_4(op1, op2, op3, op4, BODY)  \    DEF_OPC_012_4(op1, op2, op3, op4, ({BODY});)#endif /* USE_CACHE */#ifdef PREFETCH#define DISPATCH_FIRST                          \{                                               \    next_handler = pc[1].handler;               \    REDISPATCH                                  \}#define DISPATCH(level, ins_len)                \{                                               \    const void *dispatch = next_handler;        \    next_handler = (++pc)[1].handler;           \    goto *dispatch;                             \}#else#define DISPATCH_FIRST                          \    REDISPATCH#define DISPATCH(level, ins_len)                \    goto *(++pc)->handler;#endif /* PREFETCH */#define BRANCH(TEST)                            \    if(TEST) {                                  \        pc = (Instruction*)pc->operand.pntr;    \        DISPATCH_FIRST                          \    } else                                      \        DISPATCH(0,0)#define REDISPATCH                              \    goto *pc->handler;#define DISPATCH_RET(ins_len)                   \    pc++;                                       \    DISPATCH_FIRST#define DISPATCH_METHOD_RET(ins_len)            \    DISPATCH_RET(ins_len)#define DISPATCH_SWITCH                         \    DISPATCH_FIRST#define PREPARE_MB(mb)                          \    if((uintptr_t)mb->code & 0x3)               \        prepare(mb, handlers)#define ARRAY_TYPE(pc)        pc->operand.i#define SINGLE_INDEX(pc)      pc->operand.i#define DOUBLE_INDEX(pc)      pc->operand.i#define SINGLE_SIGNED(pc)     pc->operand.i#define DOUBLE_SIGNED(pc)     pc->operand.i#define IINC_LVAR_IDX(pc)     pc->operand.ii.i1#define IINC_DELTA(pc)        pc->operand.ii.i2#define INV_QUICK_ARGS(pc)    pc->operand.uu.u1#define INV_QUICK_IDX(pc)     pc->operand.uu.u2#define INV_INTF_IDX(pc)      pc->operand.uu.u1#define INV_INTF_CACHE(pc)    pc->operand.uu.u2#define MULTI_ARRAY_DIM(pc)   pc->operand.uui.u2#define RESOLVED_CONSTANT(pc) pc->operand.u#define RESOLVED_FIELD(pc)    ((FieldBlock*)pc->operand.pntr)#define RESOLVED_METHOD(pc)   ((MethodBlock*)pc->operand.pntr)#define RESOLVED_CLASS(pc)    (Class *)CP_INFO(cp, pc->operand.uui.u1)/* Macros for checking for common exceptions */#define THROW_EXCEPTION(excep_name, message)                               \{                                                                          \    frame->last_pc = pc;                                                   \    signalException(excep_name, message);                                  \    goto throwException;                                                   \}#define NULL_POINTER_CHECK(ref)                                            \    if(!ref) THROW_EXCEPTION("java/lang/NullPointerException", NULL);#define MAX_INT_DIGITS 11#define ARRAY_BOUNDS_CHECK(array, idx)                                     \{                                                                          \    if(idx >= ARRAY_LEN(array)) {                                          \        char buff[MAX_INT_DIGITS];                                         \        snprintf(buff, MAX_INT_DIGITS, "%d", idx);                         \        THROW_EXCEPTION("java/lang/ArrayIndexOutOfBoundsException", buff); \    }                                                                      \}#define ZERO_DIVISOR_CHECK(value)                                          \    if(value == 0)                                                         \        THROW_EXCEPTION("java/lang/ArithmeticException", "division by zero");extern void initialiseDirect(InitArgs *args);extern void prepare(MethodBlock *mb, const void ***handlers);

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -