2011-10-23 17:48:10 -07:00
|
|
|
#include "pyc_numeric.h"
|
2009-07-24 08:35:21 +00:00
|
|
|
#include "bytecode.h"
|
|
|
|
|
2010-11-07 15:39:36 -08:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#define snprintf _snprintf
|
|
|
|
#endif
|
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
#define DECLARE_PYTHON(maj, min) \
|
|
|
|
extern int python_##maj##min##_map(int); \
|
|
|
|
extern int python_##maj##min##_unmap(int);
|
2009-07-24 08:35:21 +00:00
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
DECLARE_PYTHON(1, 0)
|
|
|
|
DECLARE_PYTHON(1, 1)
|
|
|
|
DECLARE_PYTHON(1, 3)
|
|
|
|
DECLARE_PYTHON(1, 4)
|
|
|
|
DECLARE_PYTHON(1, 5)
|
|
|
|
DECLARE_PYTHON(1, 6)
|
|
|
|
DECLARE_PYTHON(2, 0)
|
|
|
|
DECLARE_PYTHON(2, 1)
|
|
|
|
DECLARE_PYTHON(2, 2)
|
|
|
|
DECLARE_PYTHON(2, 3)
|
|
|
|
DECLARE_PYTHON(2, 4)
|
|
|
|
DECLARE_PYTHON(2, 5)
|
|
|
|
DECLARE_PYTHON(2, 6)
|
|
|
|
DECLARE_PYTHON(2, 7)
|
|
|
|
DECLARE_PYTHON(3, 0)
|
|
|
|
DECLARE_PYTHON(3, 1)
|
2011-01-06 17:15:48 -08:00
|
|
|
DECLARE_PYTHON(3, 2)
|
2012-05-26 13:50:23 -07:00
|
|
|
DECLARE_PYTHON(3, 3)
|
2013-11-27 14:10:05 -08:00
|
|
|
DECLARE_PYTHON(3, 4)
|
2015-10-01 16:06:09 -07:00
|
|
|
DECLARE_PYTHON(3, 5)
|
2009-07-24 08:35:21 +00:00
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
const char* Pyc::OpcodeName(int opcode)
|
|
|
|
{
|
|
|
|
static const char* opcode_names[] = {
|
2011-01-06 17:15:48 -08:00
|
|
|
"STOP_CODE", "POP_TOP", "ROT_TWO", "ROT_THREE", "DUP_TOP", "DUP_TOP_TWO",
|
|
|
|
"UNARY_POSITIVE", "UNARY_NEGATIVE", "UNARY_NOT", "UNARY_CONVERT",
|
|
|
|
"UNARY_CALL", "UNARY_INVERT", "BINARY_POWER", "BINARY_MULTIPLY",
|
|
|
|
"BINARY_DIVIDE", "BINARY_MODULO", "BINARY_ADD", "BINARY_SUBTRACT",
|
|
|
|
"BINARY_SUBSCR", "BINARY_CALL", "SLICE_0", "SLICE_1", "SLICE_2", "SLICE_3",
|
|
|
|
"STORE_SLICE_0", "STORE_SLICE_1", "STORE_SLICE_2", "STORE_SLICE_3",
|
|
|
|
"DELETE_SLICE_0", "DELETE_SLICE_1", "DELETE_SLICE_2", "DELETE_SLICE_3",
|
|
|
|
"STORE_SUBSCR", "DELETE_SUBSCR", "BINARY_LSHIFT", "BINARY_RSHIFT",
|
|
|
|
"BINARY_AND", "BINARY_XOR", "BINARY_OR", "PRINT_EXPR", "PRINT_ITEM",
|
|
|
|
"PRINT_NEWLINE", "BREAK_LOOP", "RAISE_EXCEPTION", "LOAD_LOCALS",
|
|
|
|
"RETURN_VALUE", "LOAD_GLOBALS", "EXEC_STMT", "BUILD_FUNCTION", "POP_BLOCK",
|
|
|
|
"END_FINALLY", "BUILD_CLASS", "ROT_FOUR", "NOP", "LIST_APPEND",
|
|
|
|
"BINARY_FLOOR_DIVIDE", "BINARY_TRUE_DIVIDE", "INPLACE_FLOOR_DIVIDE",
|
|
|
|
"INPLACE_TRUE_DIVIDE", "STORE_MAP", "INPLACE_ADD", "INPLACE_SUBTRACT",
|
|
|
|
"INPLACE_MULTIPLY", "INPLACE_DIVIDE", "INPLACE_MODULO", "INPLACE_POWER",
|
|
|
|
"GET_ITER", "PRINT_ITEM_TO", "PRINT_NEWLINE_TO", "INPLACE_LSHIFT",
|
|
|
|
"INPLACE_RSHIFT", "INPLACE_AND", "INPLACE_XOR", "INPLACE_OR",
|
2015-10-01 16:06:09 -07:00
|
|
|
"WITH_CLEANUP", "WITH_CLEANUP_START", "WITH_CLEANUP_FINISH", "IMPORT_STAR",
|
|
|
|
"YIELD_VALUE", "LOAD_BUILD_CLASS", "STORE_LOCALS", "POP_EXCEPT", "SET_ADD",
|
|
|
|
"YIELD_FROM", "BINARY_MATRIX_MULTIPLY", "INPLACE_MATRIX_MULTIPLY",
|
|
|
|
"GET_AITER", "GET_ANEXT", "BEFORE_ASYNC_WITH", "GET_YEILD_FROM_ITER",
|
|
|
|
"GET_AWAITABLE",
|
2010-09-02 01:01:05 -07:00
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
"STORE_NAME", "DELETE_NAME", "UNPACK_TUPLE", "UNPACK_LIST", "UNPACK_ARG",
|
|
|
|
"STORE_ATTR", "DELETE_ATTR", "STORE_GLOBAL", "DELETE_GLOBAL",
|
|
|
|
"UNPACK_VARARG", "LOAD_CONST", "LOAD_NAME", "BUILD_TUPLE", "BUILD_LIST",
|
2009-07-24 08:35:21 +00:00
|
|
|
"BUILD_MAP", "LOAD_ATTR", "COMPARE_OP", "IMPORT_NAME", "IMPORT_FROM",
|
|
|
|
"JUMP_FORWARD", "JUMP_IF_FALSE", "JUMP_IF_TRUE", "JUMP_ABSOLUTE",
|
2010-09-04 01:20:41 -07:00
|
|
|
"FOR_LOOP", "LOAD_LOCAL", "LOAD_GLOBAL", "SET_FUNC_ARGS", "SETUP_LOOP",
|
|
|
|
"SETUP_EXCEPT", "SETUP_FINALLY", "RESERVE_FAST", "LOAD_FAST",
|
|
|
|
"STORE_FAST", "DELETE_FAST", "SET_LINENO", "RAISE_VARARGS",
|
|
|
|
"CALL_FUNCTION", "MAKE_FUNCTION", "BUILD_SLICE", "CALL_FUNCTION_VAR",
|
|
|
|
"CALL_FUNCTION_KW", "CALL_FUNCTION_VAR_KW", "UNPACK_SEQUENCE", "FOR_ITER",
|
|
|
|
"DUP_TOPX", "BUILD_SET", "JUMP_IF_FALSE_OR_POP", "JUMP_IF_TRUE_OR_POP",
|
|
|
|
"POP_JUMP_IF_FALSE", "POP_JUMP_IF_TRUE", "CONTINUE_LOOP", "MAKE_CLOSURE",
|
2011-01-06 17:15:48 -08:00
|
|
|
"LOAD_CLOSURE", "LOAD_DEREF", "STORE_DEREF", "DELETE_DEREF",
|
|
|
|
"EXTENDED_ARG", "SETUP_WITH", "SET_ADD", "MAP_ADD", "UNPACK_EX",
|
2015-10-01 16:06:09 -07:00
|
|
|
"LIST_APPEND", "LOAD_CLASSDEREF", "BUILD_LIST_UNPACK", "BUILD_MAP_UNPACK",
|
|
|
|
"BUILD_MAP_UNPACK_WITH_CALL", "BUILD_TUPLE_UNPACK", "BUILD_SET_UNPACK",
|
|
|
|
"SETUP_ASYNC_WITH",
|
2010-09-04 01:20:41 -07:00
|
|
|
};
|
2009-07-24 08:35:21 +00:00
|
|
|
|
2015-10-01 16:06:09 -07:00
|
|
|
#if __cplusplus >= 201103L
|
|
|
|
static_assert(sizeof(opcode_names) / sizeof(opcode_names[0]) == PYC_LAST_OPCODE,
|
|
|
|
"Pyc::OpcodeName opcode_names not in sync with opcode enum");
|
|
|
|
#endif
|
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
if (opcode < 0)
|
|
|
|
return "<INVALID>";
|
2009-07-24 19:52:47 +00:00
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
if (opcode < PYC_LAST_OPCODE)
|
|
|
|
return opcode_names[opcode];
|
2009-07-24 19:52:47 +00:00
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
static char badcode[10];
|
|
|
|
snprintf(badcode, 10, "<%d>", opcode);
|
|
|
|
return badcode;
|
|
|
|
};
|
2009-07-24 19:52:47 +00:00
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
int Pyc::ByteToOpcode(int maj, int min, int opcode)
|
2009-07-24 19:52:47 +00:00
|
|
|
{
|
2010-09-04 01:20:41 -07:00
|
|
|
switch (maj) {
|
|
|
|
case 1:
|
|
|
|
switch (min) {
|
|
|
|
case 0: return python_10_map(opcode);
|
|
|
|
case 1: return python_11_map(opcode);
|
|
|
|
case 3: return python_13_map(opcode);
|
|
|
|
case 4: return python_14_map(opcode);
|
|
|
|
case 5: return python_15_map(opcode);
|
|
|
|
case 6: return python_16_map(opcode);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
switch (min) {
|
|
|
|
case 0: return python_20_map(opcode);
|
|
|
|
case 1: return python_21_map(opcode);
|
|
|
|
case 2: return python_22_map(opcode);
|
|
|
|
case 3: return python_23_map(opcode);
|
|
|
|
case 4: return python_24_map(opcode);
|
|
|
|
case 5: return python_25_map(opcode);
|
|
|
|
case 6: return python_26_map(opcode);
|
|
|
|
case 7: return python_27_map(opcode);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
switch (min) {
|
|
|
|
case 0: return python_30_map(opcode);
|
|
|
|
case 1: return python_31_map(opcode);
|
2011-01-06 17:15:48 -08:00
|
|
|
case 2: return python_32_map(opcode);
|
2012-05-26 13:50:23 -07:00
|
|
|
case 3: return python_33_map(opcode);
|
2013-11-27 14:10:05 -08:00
|
|
|
case 4: return python_34_map(opcode);
|
2015-10-01 16:06:09 -07:00
|
|
|
case 5: return python_35_map(opcode);
|
2010-09-04 01:20:41 -07:00
|
|
|
}
|
|
|
|
break;
|
2010-09-02 01:01:05 -07:00
|
|
|
}
|
2010-09-04 01:20:41 -07:00
|
|
|
return PYC_INVALID_OPCODE;
|
2009-07-24 19:52:47 +00:00
|
|
|
}
|
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
bool Pyc::IsConstArg(int opcode)
|
2009-07-24 19:52:47 +00:00
|
|
|
{
|
2010-09-04 01:20:41 -07:00
|
|
|
return (opcode == Pyc::LOAD_CONST_A) || (opcode == Pyc::RESERVE_FAST_A);
|
2009-07-24 19:52:47 +00:00
|
|
|
}
|
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
bool Pyc::IsNameArg(int opcode)
|
2010-08-31 23:17:38 -07:00
|
|
|
{
|
2010-09-04 01:20:41 -07:00
|
|
|
return (opcode == Pyc::DELETE_ATTR_A) || (opcode == Pyc::DELETE_GLOBAL_A) ||
|
|
|
|
(opcode == Pyc::DELETE_NAME_A) || (opcode == Pyc::IMPORT_FROM_A) ||
|
|
|
|
(opcode == Pyc::IMPORT_NAME_A) || (opcode == Pyc::LOAD_ATTR_A) ||
|
|
|
|
(opcode == Pyc::LOAD_GLOBAL_A) || (opcode == Pyc::LOAD_LOCAL_A) ||
|
|
|
|
(opcode == Pyc::LOAD_NAME_A) || (opcode == Pyc::STORE_ATTR_A) ||
|
|
|
|
(opcode == Pyc::STORE_GLOBAL_A) || (opcode == Pyc::STORE_NAME_A);
|
2010-08-31 23:17:38 -07:00
|
|
|
}
|
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
bool Pyc::IsVarNameArg(int opcode)
|
2009-07-24 19:52:47 +00:00
|
|
|
{
|
2010-09-04 01:20:41 -07:00
|
|
|
return (opcode == Pyc::DELETE_FAST_A) || (opcode == Pyc::LOAD_FAST_A) ||
|
|
|
|
(opcode == Pyc::STORE_FAST_A);
|
2009-07-24 19:52:47 +00:00
|
|
|
}
|
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
bool Pyc::IsCellArg(int opcode)
|
2009-07-24 19:52:47 +00:00
|
|
|
{
|
2010-09-04 01:20:41 -07:00
|
|
|
return (opcode == Pyc::LOAD_CLOSURE_A) || (opcode == Pyc::LOAD_DEREF_A) ||
|
|
|
|
(opcode == Pyc::STORE_DEREF_A);
|
2009-07-24 19:52:47 +00:00
|
|
|
}
|
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
bool Pyc::IsJumpOffsetArg(int opcode)
|
2009-07-24 19:52:47 +00:00
|
|
|
{
|
2010-09-04 01:20:41 -07:00
|
|
|
return (opcode == Pyc::JUMP_FORWARD_A) || (opcode == Pyc::JUMP_IF_FALSE_A) ||
|
2010-12-31 02:42:58 -08:00
|
|
|
(opcode == Pyc::JUMP_IF_TRUE_A) || (opcode == Pyc::SETUP_LOOP_A) ||
|
2011-01-01 02:31:31 -08:00
|
|
|
(opcode == Pyc::SETUP_FINALLY_A) || (opcode == Pyc::SETUP_EXCEPT_A) ||
|
|
|
|
(opcode == Pyc::FOR_LOOP_A);
|
2010-08-31 23:17:38 -07:00
|
|
|
}
|
|
|
|
|
2009-07-24 19:52:47 +00:00
|
|
|
|
2009-07-26 10:07:13 +00:00
|
|
|
void print_const(PycRef<PycObject> obj, PycModule* mod)
|
2009-07-24 19:52:47 +00:00
|
|
|
{
|
|
|
|
switch (obj->type()) {
|
|
|
|
case PycObject::TYPE_STRING:
|
2014-01-21 00:47:11 -08:00
|
|
|
OutputString(obj.cast<PycString>(), (mod->majorVer() == 3) ? 'b' : 0);
|
|
|
|
break;
|
|
|
|
case PycObject::TYPE_UNICODE:
|
|
|
|
OutputString(obj.cast<PycString>(), (mod->majorVer() == 3) ? 0 : 'u');
|
|
|
|
break;
|
2009-07-24 19:52:47 +00:00
|
|
|
case PycObject::TYPE_STRINGREF:
|
|
|
|
case PycObject::TYPE_INTERNED:
|
2014-01-21 00:07:34 -08:00
|
|
|
case PycObject::TYPE_ASCII:
|
|
|
|
case PycObject::TYPE_ASCII_INTERNED:
|
|
|
|
case PycObject::TYPE_SHORT_ASCII:
|
|
|
|
case PycObject::TYPE_SHORT_ASCII_INTERNED:
|
2014-01-21 00:47:11 -08:00
|
|
|
OutputString(obj.cast<PycString>(), 0);
|
2009-07-25 00:02:31 +00:00
|
|
|
break;
|
2009-07-24 19:52:47 +00:00
|
|
|
case PycObject::TYPE_TUPLE:
|
2014-01-21 00:07:34 -08:00
|
|
|
case PycObject::TYPE_SMALL_TUPLE:
|
2009-07-24 19:52:47 +00:00
|
|
|
{
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "(");
|
2009-07-24 19:52:47 +00:00
|
|
|
PycTuple::value_t values = obj.cast<PycTuple>()->values();
|
2009-07-27 00:23:49 +00:00
|
|
|
PycTuple::value_t::const_iterator it = values.begin();
|
2009-07-24 19:52:47 +00:00
|
|
|
if (it != values.end()) {
|
2009-07-25 00:02:31 +00:00
|
|
|
print_const(*it, mod);
|
2009-07-24 19:52:47 +00:00
|
|
|
while (++it != values.end()) {
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, ", ");
|
2009-07-25 00:02:31 +00:00
|
|
|
print_const(*it, mod);
|
2009-07-24 19:52:47 +00:00
|
|
|
}
|
|
|
|
}
|
2009-07-27 03:00:55 +00:00
|
|
|
if (values.size() == 1)
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, ",)");
|
2009-07-27 03:00:55 +00:00
|
|
|
else
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, ")");
|
2009-07-24 19:52:47 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case PycObject::TYPE_LIST:
|
|
|
|
{
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "[");
|
2009-07-24 19:52:47 +00:00
|
|
|
PycList::value_t values = obj.cast<PycList>()->values();
|
2009-07-27 00:23:49 +00:00
|
|
|
PycList::value_t::const_iterator it = values.begin();
|
2009-07-24 19:52:47 +00:00
|
|
|
if (it != values.end()) {
|
2009-07-25 00:02:31 +00:00
|
|
|
print_const(*it, mod);
|
2009-07-24 19:52:47 +00:00
|
|
|
while (++it != values.end()) {
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, ", ");
|
2009-07-25 00:02:31 +00:00
|
|
|
print_const(*it, mod);
|
2009-07-24 19:52:47 +00:00
|
|
|
}
|
|
|
|
}
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "]");
|
2009-07-24 19:52:47 +00:00
|
|
|
}
|
|
|
|
break;
|
2009-07-24 21:39:51 +00:00
|
|
|
case PycObject::TYPE_DICT:
|
|
|
|
{
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "{");
|
2009-07-24 21:39:51 +00:00
|
|
|
PycDict::key_t keys = obj.cast<PycDict>()->keys();
|
|
|
|
PycDict::value_t values = obj.cast<PycDict>()->values();
|
2009-07-27 00:23:49 +00:00
|
|
|
PycDict::key_t::const_iterator ki = keys.begin();
|
|
|
|
PycDict::value_t::const_iterator vi = values.begin();
|
2009-07-24 21:39:51 +00:00
|
|
|
if (ki != keys.end()) {
|
2009-07-25 00:02:31 +00:00
|
|
|
print_const(*ki, mod);
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, ": ");
|
2009-07-25 00:02:31 +00:00
|
|
|
print_const(*vi, mod);
|
2009-07-24 21:39:51 +00:00
|
|
|
while (++ki != keys.end()) {
|
|
|
|
++vi;
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, ", ");
|
2009-07-25 00:02:31 +00:00
|
|
|
print_const(*ki, mod);
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, ": ");
|
2009-07-25 00:02:31 +00:00
|
|
|
print_const(*vi, mod);
|
2009-07-24 21:39:51 +00:00
|
|
|
}
|
|
|
|
}
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "}");
|
2009-07-24 21:39:51 +00:00
|
|
|
}
|
|
|
|
break;
|
2009-07-25 02:41:15 +00:00
|
|
|
case PycObject::TYPE_SET:
|
|
|
|
{
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "{");
|
2009-07-25 02:41:15 +00:00
|
|
|
PycSet::value_t values = obj.cast<PycSet>()->values();
|
2009-07-27 00:23:49 +00:00
|
|
|
PycSet::value_t::const_iterator it = values.begin();
|
2009-07-25 02:41:15 +00:00
|
|
|
if (it != values.end()) {
|
|
|
|
print_const(*it, mod);
|
|
|
|
while (++it != values.end()) {
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, ", ");
|
2009-07-25 02:41:15 +00:00
|
|
|
print_const(*it, mod);
|
|
|
|
}
|
|
|
|
}
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "}");
|
2009-07-25 02:41:15 +00:00
|
|
|
}
|
|
|
|
break;
|
2009-07-24 19:52:47 +00:00
|
|
|
case PycObject::TYPE_NONE:
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "None");
|
2009-07-24 19:52:47 +00:00
|
|
|
break;
|
|
|
|
case PycObject::TYPE_TRUE:
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "True");
|
2009-07-24 19:52:47 +00:00
|
|
|
break;
|
|
|
|
case PycObject::TYPE_FALSE:
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "False");
|
2009-07-24 19:52:47 +00:00
|
|
|
break;
|
|
|
|
case PycObject::TYPE_INT:
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "%d", obj.cast<PycInt>()->value());
|
2009-07-24 19:52:47 +00:00
|
|
|
break;
|
2011-09-23 21:46:05 -07:00
|
|
|
case PycObject::TYPE_LONG:
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "%s", obj.cast<PycLong>()->repr().c_str());
|
2011-09-23 21:46:05 -07:00
|
|
|
break;
|
2009-07-24 19:52:47 +00:00
|
|
|
case PycObject::TYPE_FLOAT:
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "%s", obj.cast<PycFloat>()->value());
|
2009-07-24 19:52:47 +00:00
|
|
|
break;
|
2009-07-25 02:41:15 +00:00
|
|
|
case PycObject::TYPE_COMPLEX:
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "(%s+%sj)", obj.cast<PycComplex>()->value(),
|
|
|
|
obj.cast<PycComplex>()->imag());
|
2009-07-25 02:41:15 +00:00
|
|
|
break;
|
|
|
|
case PycObject::TYPE_BINARY_FLOAT:
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "%g", obj.cast<PycCFloat>()->value());
|
2009-07-25 02:41:15 +00:00
|
|
|
break;
|
|
|
|
case PycObject::TYPE_BINARY_COMPLEX:
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "(%g+%gj)", obj.cast<PycCComplex>()->value(),
|
|
|
|
obj.cast<PycCComplex>()->imag());
|
2009-07-25 02:41:15 +00:00
|
|
|
break;
|
2009-07-24 19:52:47 +00:00
|
|
|
case PycObject::TYPE_CODE:
|
2009-07-25 00:43:46 +00:00
|
|
|
case PycObject::TYPE_CODE2:
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "<CODE> %s", obj.cast<PycCode>()->name()->value());
|
2009-07-24 19:52:47 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-26 10:07:13 +00:00
|
|
|
void bc_next(PycBuffer& source, PycModule* mod, int& opcode, int& operand, int& pos)
|
|
|
|
{
|
2010-09-04 01:20:41 -07:00
|
|
|
opcode = Pyc::ByteToOpcode(mod->majorVer(), mod->minorVer(), source.getByte());
|
2009-07-26 10:07:13 +00:00
|
|
|
operand = 0;
|
|
|
|
pos += 1;
|
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
if (opcode == Pyc::EXTENDED_ARG_A) {
|
2009-07-26 10:07:13 +00:00
|
|
|
operand = source.get16() << 16;
|
2013-08-06 22:16:28 -07:00
|
|
|
opcode = Pyc::ByteToOpcode(mod->majorVer(), mod->minorVer(), source.getByte());
|
2009-07-26 10:07:13 +00:00
|
|
|
pos += 3;
|
|
|
|
}
|
2010-09-04 01:20:41 -07:00
|
|
|
if (opcode >= Pyc::PYC_HAVE_ARG) {
|
2013-08-06 22:16:28 -07:00
|
|
|
operand |= source.get16();
|
2009-07-26 10:07:13 +00:00
|
|
|
pos += 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-24 08:35:21 +00:00
|
|
|
void bc_disasm(PycRef<PycCode> code, PycModule* mod, int indent)
|
|
|
|
{
|
|
|
|
PycBuffer source(code->code()->value(), code->code()->length());
|
|
|
|
|
2009-07-26 10:07:13 +00:00
|
|
|
int opcode, operand;
|
2009-07-25 02:41:15 +00:00
|
|
|
int pos = 0;
|
2009-07-24 08:35:21 +00:00
|
|
|
while (!source.atEof()) {
|
2009-07-25 02:41:15 +00:00
|
|
|
for (int i=0; i<indent; i++)
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, " ");
|
|
|
|
fprintf(pyc_output, "%-7d ", pos); // Current bytecode position
|
2009-07-25 02:41:15 +00:00
|
|
|
|
2009-07-26 10:07:13 +00:00
|
|
|
bc_next(source, mod, opcode, operand, pos);
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "%-24s", Pyc::OpcodeName(opcode));
|
2009-07-24 08:35:21 +00:00
|
|
|
|
2010-09-04 01:20:41 -07:00
|
|
|
if (opcode >= Pyc::PYC_HAVE_ARG) {
|
|
|
|
if (Pyc::IsConstArg(opcode)) {
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "%d: ", operand);
|
2009-07-25 00:02:31 +00:00
|
|
|
print_const(code->getConst(operand), mod);
|
2010-09-04 01:20:41 -07:00
|
|
|
} else if (Pyc::IsNameArg(opcode)) {
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "%d: %s", operand, code->getName(operand)->value());
|
2010-09-04 01:20:41 -07:00
|
|
|
} else if (Pyc::IsVarNameArg(opcode)) {
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "%d: %s", operand, code->getVarName(operand)->value());
|
2010-09-04 01:20:41 -07:00
|
|
|
} else if (Pyc::IsCellArg(opcode)) {
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "%d: ", operand);
|
2011-10-16 22:46:17 -07:00
|
|
|
print_const(code->getCellVar(operand), mod);
|
2010-09-04 01:20:41 -07:00
|
|
|
} else if (Pyc::IsJumpOffsetArg(opcode)) {
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "%d (to %d)", operand, pos+operand);
|
2009-07-24 19:52:47 +00:00
|
|
|
} else {
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "%d", operand);
|
2009-07-24 19:52:47 +00:00
|
|
|
}
|
|
|
|
}
|
2011-10-23 19:04:06 -07:00
|
|
|
fprintf(pyc_output, "\n");
|
2009-07-24 08:35:21 +00:00
|
|
|
}
|
|
|
|
}
|