You've already forked python-uncompyle6
mirror of
https://github.com/rocky/python-uncompyle6.git
synced 2025-08-02 16:44:46 +08:00
Not working: sync with decompyle3
This commit is contained in:
@@ -21,6 +21,7 @@ scanner/ingestion module. From here we call various version-specific
|
||||
scanners, e.g. for Python 2.7 or 3.4.
|
||||
"""
|
||||
|
||||
from abc import ABC
|
||||
from array import array
|
||||
from collections import namedtuple
|
||||
from types import ModuleType
|
||||
@@ -89,7 +90,7 @@ def long(num):
|
||||
CONST_COLLECTIONS = ("CONST_LIST", "CONST_SET", "CONST_DICT", "CONST_MAP")
|
||||
|
||||
|
||||
class Code(object):
|
||||
class Code:
|
||||
"""
|
||||
Class for representing code-objects.
|
||||
|
||||
@@ -108,7 +109,7 @@ class Code(object):
|
||||
self._tokens, self._customize = scanner.ingest(co, classname, show_asm=show_asm)
|
||||
|
||||
|
||||
class Scanner:
|
||||
class Scanner(ABC):
|
||||
def __init__(self, version: tuple, show_asm=None, is_pypy=False):
|
||||
self.version = version
|
||||
self.show_asm = show_asm
|
||||
@@ -293,18 +294,51 @@ class Scanner:
|
||||
return False
|
||||
return offset < self.get_target(offset)
|
||||
|
||||
def ingest(self, co, classname=None, code_objects={}, show_asm=None):
|
||||
"""
|
||||
Code to tokenize disassembly. Subclasses must implement this.
|
||||
"""
|
||||
raise NotImplementedError("This method should have been implemented")
|
||||
|
||||
def prev_offset(self, offset: int) -> int:
|
||||
return self.insts[self.offset2inst_index[offset] - 1].offset
|
||||
|
||||
def get_inst(self, offset: int):
|
||||
# Instructions can get moved as a result of EXTENDED_ARGS removal.
|
||||
# So if "offset" is not in self.offset2inst_index, then
|
||||
# we assume that it was an instruction moved back.
|
||||
# We check that assumption though by looking at
|
||||
# self.code's opcode.
|
||||
"""
|
||||
Returns the instruction from ``self.insts`` that has at offset
|
||||
``offset``.
|
||||
|
||||
Instructions can get moved as a result of ``EXTENDED_ARGS`` removal.
|
||||
So if ``offset`` is not in self.offset2inst_index, then
|
||||
we assume that it was an instruction moved back.
|
||||
We check that assumption though by looking at
|
||||
self.code's opcode.
|
||||
Sadly instructions can get moved forward too.
|
||||
So we have to check which direction we are going.
|
||||
"""
|
||||
offset_increment = instruction_size(self.opc.EXTENDED_ARG, self.opc)
|
||||
if offset not in self.offset2inst_index:
|
||||
offset -= instruction_size(self.opc.EXTENDED_ARG, self.opc)
|
||||
assert self.code[offset] == self.opc.EXTENDED_ARG
|
||||
if self.code[offset] != self.opc.EXTENDED_ARG:
|
||||
target_name = self.opc.opname[self.code[offset]]
|
||||
# JUMP_ABSOLUTE can be like this where
|
||||
# the inst offset is at what used to be an EXTENDED_ARG
|
||||
# so find the first extended arg.
|
||||
next_offset = offset - offset_increment
|
||||
while next_offset not in self.offset2inst_index:
|
||||
next_offset -= offset_increment
|
||||
assert self.code[next_offset] == self.opc.EXTENDED_ARG
|
||||
inst = self.insts[self.offset2inst_index[next_offset]]
|
||||
assert inst.opname == target_name, inst
|
||||
else:
|
||||
next_offset = offset + offset_increment
|
||||
while next_offset not in self.offset2inst_index:
|
||||
next_offset += offset_increment
|
||||
|
||||
inst = self.insts[self.offset2inst_index[next_offset]]
|
||||
|
||||
assert inst.has_extended_arg is True
|
||||
return inst
|
||||
|
||||
return self.insts[self.offset2inst_index[offset]]
|
||||
|
||||
def get_target(self, offset: int, extended_arg: int = 0) -> int:
|
||||
@@ -446,7 +480,7 @@ class Scanner:
|
||||
pass
|
||||
pass
|
||||
pass
|
||||
if inst.offset >= end:
|
||||
if isinstance(inst, int) and inst.offset >= end:
|
||||
break
|
||||
pass
|
||||
|
||||
|
@@ -53,7 +53,7 @@ class Scanner37(Scanner37Base):
|
||||
if collection_type == "CONST_DICT":
|
||||
# constant dictionaries work via BUILD_CONST_KEY_MAP and
|
||||
# handle the values() like sets and lists.
|
||||
# However the keys() are an LOAD_CONST of the keys.
|
||||
# However, the keys() are an LOAD_CONST of the keys.
|
||||
# adjust offset to account for this
|
||||
count += 1
|
||||
|
||||
|
@@ -268,7 +268,7 @@ class Scanner37Base(Scanner):
|
||||
and next_inst.argval == "AssertionError"
|
||||
and inst.argval
|
||||
):
|
||||
raise_idx = self.offset2inst_index[self.prev_op[inst.argval]]
|
||||
raise_idx = self.get_inst(self.prev_op[inst.argval])
|
||||
raise_inst = self.insts[raise_idx]
|
||||
if raise_inst.opname.startswith("RAISE_VARARGS"):
|
||||
self.load_asserts.add(next_inst.offset)
|
||||
@@ -286,7 +286,7 @@ class Scanner37Base(Scanner):
|
||||
# some backward jumps, are turned into forward jumps to another
|
||||
# "extended arg" backward jump to the same location.
|
||||
if inst.opname == "JUMP_FORWARD":
|
||||
jump_inst = self.insts[self.offset2inst_index[inst.argval]]
|
||||
jump_inst = self.get_inst(inst.argval)
|
||||
if jump_inst.has_extended_arg and jump_inst.opname.startswith("JUMP"):
|
||||
# Create a combination of the jump-to instruction and
|
||||
# this one. Keep the position information of this instruction,
|
||||
|
Reference in New Issue
Block a user