mirror of
https://github.com/cemu-project/idapython.git
synced 2025-01-15 03:19:20 +01:00
8e78513e91
- updated CHANGES.txt - nalt.i: forgot to release reference for callback result (in py_import_enum_cb())
229 lines
7.3 KiB
Python
229 lines
7.3 KiB
Python
#!/usr/bin/env python
|
|
# -----------------------------------------------------------------------
|
|
# IDAPython - Python plugin for Interactive Disassembler Pro
|
|
#
|
|
# Copyright (c) 2004-2009 Gergely Erdelyi <dyce@d-dome.net>
|
|
#
|
|
# All rights reserved.
|
|
#
|
|
# For detailed copyright information see the file COPYING in
|
|
# the root of the distribution archive.
|
|
# -----------------------------------------------------------------------
|
|
# init.py - Essential init routines
|
|
# -----------------------------------------------------------------------
|
|
import os
|
|
import sys
|
|
import time
|
|
import warnings
|
|
|
|
import _idaapi
|
|
|
|
# __EA64__ is set if IDA is running in 64-bit mode
|
|
__EA64__ = _idaapi.BADADDR == 0xFFFFFFFFFFFFFFFFL
|
|
|
|
# -----------------------------------------------------------------------
|
|
def addscriptpath(script):
|
|
"""
|
|
Add the path part of the scriptfile to the system path to
|
|
allow modules to be loaded from the same place.
|
|
|
|
Each path is added only once.
|
|
"""
|
|
pathfound = 0
|
|
|
|
scriptpath = os.path.dirname(script)
|
|
|
|
for pathitem in sys.path:
|
|
if pathitem == scriptpath:
|
|
pathfound = 1
|
|
break
|
|
|
|
if pathfound == 0:
|
|
sys.path.append(scriptpath)
|
|
|
|
# Add the script to ScriptBox if it's not there yet
|
|
if not script in scriptbox.list:
|
|
scriptbox.list.insert(0, script)
|
|
|
|
# ------------------------------------------------------------
|
|
def runscript(script):
|
|
"""
|
|
Run the specified script after adding its directory path to
|
|
system path.
|
|
|
|
This function is used by the low-level plugin code.
|
|
"""
|
|
addscriptpath(script)
|
|
watchdog.reset()
|
|
# Save the argv, path, I/O and base modules for later cleanup
|
|
argv = sys.argv
|
|
path = sys.path
|
|
stdio = [sys.stdin, sys.stdout, sys.stderr]
|
|
basemodules = sys.modules.copy()
|
|
sys.argv = [ script ]
|
|
# Adjust the __file__ path in the globals we pass to the script
|
|
g = globals()
|
|
old__file__ = g['__file__'] if '__file__' in g else ''
|
|
g['__file__'] = script
|
|
try:
|
|
execfile(script, g)
|
|
except:
|
|
raise
|
|
finally:
|
|
# Restore the globals to the state before the script was run
|
|
g['__file__'] = old__file__
|
|
sys.argv = argv
|
|
sys.path = path
|
|
sys.stdin, sys.stdout, sys.stderr = stdio
|
|
# Clean up the modules loaded by the script
|
|
for module in sys.modules.keys():
|
|
if not module in basemodules:
|
|
del(sys.modules[module])
|
|
|
|
# -----------------------------------------------------------------------
|
|
def print_banner():
|
|
banner = [
|
|
"Python interpreter version %d.%d.%d %s (serial %d)" % sys.version_info,
|
|
"Copyright (c) 1990-2009 Python Software Foundation - http://www.python.org/",
|
|
"",
|
|
"IDAPython" + (" 64-bit" if __EA64__ else "") + " version %d.%d.%d %s (serial %d)" % IDAPYTHON_VERSION,
|
|
"Copyright (c) 2004-2009 Gergely Erdelyi - http://d-dome.net/idapython/"
|
|
]
|
|
sepline = '-' * max([len(s) for s in banner])
|
|
|
|
print sepline
|
|
print "\n".join(banner)
|
|
print sepline
|
|
|
|
# -----------------------------------------------------------------------
|
|
# Take over the standard text outputs
|
|
# -----------------------------------------------------------------------
|
|
class MyStdOut:
|
|
"""
|
|
Dummy file-like class that receives stout and stderr
|
|
"""
|
|
def write(self, text):
|
|
# Swap out the unprintable characters
|
|
text = text.decode('ascii', 'replace').encode('ascii', 'replace')
|
|
# Print to IDA message window
|
|
_idaapi.msg(text.replace("%", "%%"))
|
|
|
|
def flush(self):
|
|
pass
|
|
|
|
def isatty(self):
|
|
return False
|
|
|
|
# Redirect stderr and stdout to the IDA message window
|
|
sys.stdout = sys.stderr = MyStdOut()
|
|
|
|
# Assign a default sys.argv
|
|
sys.argv = [""]
|
|
|
|
# Have to make sure Python finds our modules
|
|
sys.path.append(_idaapi.idadir("python"))
|
|
|
|
# -----------------------------------------------------------------------
|
|
# Import all the required modules
|
|
# -----------------------------------------------------------------------
|
|
from idaapi import Choose, get_user_idadir, cvar, Choose2, Appcall
|
|
from idc import *
|
|
from idautils import *
|
|
import idaapi
|
|
|
|
# -----------------------------------------------------------------------
|
|
# Build up the ScriptBox tool
|
|
# -----------------------------------------------------------------------
|
|
class ScriptBox(Choose):
|
|
def __init__(self, list=None):
|
|
if list:
|
|
self.list = list
|
|
else:
|
|
self.list = []
|
|
Choose.__init__(self, self.list, "ScriptBox", 1)
|
|
self.width = 50
|
|
|
|
def run(self):
|
|
if len(self.list) == 0:
|
|
Warning("ScriptBox history is empty.\nRun some script with Alt-9 and try again.")
|
|
return None
|
|
|
|
n = self.choose()
|
|
|
|
if n > 0:
|
|
runscript(self.list[n-1])
|
|
|
|
def addscript(self, scriptpath):
|
|
self.list.append(scriptpath)
|
|
|
|
scriptbox = ScriptBox()
|
|
|
|
# -----------------------------------------------------------------------
|
|
# Watchdog to catch runaway scripts after a specified timeout
|
|
#
|
|
# Usage:
|
|
# watchdog.install()
|
|
# watchdog.activate(10) # Use 10-second timeout
|
|
#
|
|
# Note: The watchdog only works for code running inside
|
|
# functions, not in global/module namespace.
|
|
# -----------------------------------------------------------------------
|
|
class WatchDog():
|
|
"""
|
|
Python tracer-based watchdog class
|
|
"""
|
|
def __init__(self, timeout=10):
|
|
self.timestamp = 0
|
|
self.timeout = timeout
|
|
self.installed = False
|
|
self.active = False
|
|
|
|
def install(self):
|
|
""" Install the tracer function, required for the watchdog """
|
|
if not self.installed:
|
|
sys.settrace(self.tracer)
|
|
self.installed = True
|
|
|
|
def activate(self, timeout=None):
|
|
""" Activate the watchdog, with optional timeout change """
|
|
assert self.installed, "WatchDog must be installed before activating"
|
|
if timeout:
|
|
self.timeout = timeout
|
|
self.reset()
|
|
self.active = True
|
|
|
|
def deactivate(self):
|
|
""" Deactivate the watchdog """
|
|
self.active = True
|
|
|
|
def reset(self):
|
|
""" Reset the timer, useful for long-running scripts """
|
|
self.timestamp = time.clock()
|
|
|
|
def tracer(self, frame, event, arg):
|
|
""" Tracer function that receives the tracing events """
|
|
if not self.active:
|
|
return None
|
|
if event == 'line':
|
|
if time.clock() - self.timestamp > self.timeout:
|
|
if AskYN(0, "The script has not finished in %d seconds\nWould you like to stop it now?" % self.timeout) == 1:
|
|
raise KeyboardInterrupt
|
|
else:
|
|
self.timestamp = time.clock()
|
|
return self.tracer
|
|
|
|
watchdog = WatchDog(10)
|
|
|
|
# -----------------------------------------------------------------------
|
|
# Load the users personal init file
|
|
userrc = get_user_idadir() + os.sep + "idapythonrc.py"
|
|
|
|
# -----------------------------------------------------------------------
|
|
if os.path.exists(userrc):
|
|
runscript(userrc)
|
|
|
|
# Remove the user script from the history
|
|
del scriptbox.list[0]
|
|
|
|
# All done, ready to rock.
|