mirror of
https://github.com/cemu-project/idapython.git
synced 2025-01-14 10:59:13 +01:00
223e4a5268
- added MakeYword - wrap obsolete IDA API functions too, so that old scripts continue to work - added debugger trace related functions in idc.py BUGFIX: IDAPython would fail with a cryptic error message if there was no free space on the current disk BUGFIX: site-packages directory was missing from sys.path BUGFIX: functions for working with additional lines (LineA/LineB etc) were broken BUGFIX: GetFixup* functions from idc.py were broken BUGFIX: IDA could not start if another python27.dll was present in PATH and importing site.py for that Python installation failed BUGFIX: SaveBase() was broken BUGFIX: IDAPython would fail with a cryptic error message if there was no free space on the current disk BUGFIX: IDAPython: site-packages directory was missing from sys.path BUGFIX: IDAPython: functions for working with additional lines (LineA/LineB etc) were broken BUGFIX: IDAPython: GetFixup* functions from idc.py were broken BUGFIX: IDAPython: IDA could not start if another python27.dll was present in PATH and importing site.py for that Python installation failed BUGFIX: IDAPython: SaveBase() was broken BUGFIX: Dbg/Idp hooks automatically unhook when hook object is not referenced anymore note: these patches are already applied in IDAPython that comes with IDA Pro 6.4
236 lines
8.4 KiB
C++
236 lines
8.4 KiB
C++
#ifndef __PYWRAPS_HPP__
|
|
#define __PYWRAPS_HPP__
|
|
|
|
//------------------------------------------------------------------------
|
|
// Types
|
|
#ifndef PYUL_DEFINED
|
|
#define PYUL_DEFINED
|
|
typedef unsigned PY_LONG_LONG PY_ULONG_LONG;
|
|
#ifdef __EA64__
|
|
typedef unsigned PY_LONG_LONG pyul_t;
|
|
typedef PY_LONG_LONG pyl_t;
|
|
#else
|
|
typedef unsigned long pyul_t;
|
|
typedef long pyl_t;
|
|
#endif
|
|
#endif
|
|
|
|
#ifdef __EA64__
|
|
#define PY_FMT64 "K"
|
|
#define PY_SFMT64 "L"
|
|
#else
|
|
#define PY_FMT64 "k"
|
|
#define PY_SFMT64 "l"
|
|
#endif
|
|
|
|
//------------------------------------------------------------------------
|
|
#define S_IDAAPI_MODNAME "idaapi"
|
|
#define S_IDC_MODNAME "idc"
|
|
#define S_IDAAPI_EXECSCRIPT "IDAPython_ExecScript"
|
|
#define S_IDAAPI_COMPLETION "IDAPython_Completion"
|
|
#define S_IDAAPI_FORMATEXC "IDAPython_FormatExc"
|
|
|
|
//------------------------------------------------------------------------
|
|
// Vector of PyObject*
|
|
typedef qvector<PyObject *> ppyobject_vec_t;
|
|
|
|
//------------------------------------------------------------------------
|
|
// PyIdc conversion object IDs
|
|
#define PY_ICID_INT64 0
|
|
#define PY_ICID_BYREF 1
|
|
#define PY_ICID_OPAQUE 2
|
|
|
|
//------------------------------------------------------------------------
|
|
// Constants used with the notify_when()
|
|
#define NW_OPENIDB 0x0001
|
|
#define NW_OPENIDB_SLOT 0
|
|
#define NW_CLOSEIDB 0x0002
|
|
#define NW_CLOSEIDB_SLOT 1
|
|
#define NW_INITIDA 0x0004
|
|
#define NW_INITIDA_SLOT 2
|
|
#define NW_TERMIDA 0x0008
|
|
#define NW_TERMIDA_SLOT 3
|
|
#define NW_REMOVE 0x0010 // Uninstall flag
|
|
#define NW_EVENTSCNT 4 // Count of notify_when codes
|
|
|
|
//------------------------------------------------------------------------
|
|
// Constants used by the pyvar_to_idcvar and idcvar_to_pyvar functions
|
|
#define CIP_FAILED -1 // Conversion error
|
|
#define CIP_IMMUTABLE 0 // Immutable object passed. Will not update the object but no error occured
|
|
#define CIP_OK 1 // Success
|
|
#define CIP_OK_NODECREF 2 // Success but do not decrement its reference
|
|
|
|
//---------------------------------------------------------------------------
|
|
// Helper macro to create C counterparts of Python py_clinked_object_t object
|
|
#ifdef __PYWRAPS__
|
|
#define DECLARE_PY_CLINKED_OBJECT(type) \
|
|
static PyObject *type##_create() \
|
|
{ \
|
|
return PyCObject_FromVoidPtr(new type(), NULL); \
|
|
} \
|
|
static bool type##_destroy(PyObject *py_obj) \
|
|
{ \
|
|
if ( !PyCObject_Check(py_obj) ) \
|
|
return false; \
|
|
delete (type *)PyCObject_AsVoidPtr(py_obj); \
|
|
return true; \
|
|
} \
|
|
static type *type##_get_clink(PyObject *self) \
|
|
{ \
|
|
return (type *)pyobj_get_clink(self); \
|
|
} \
|
|
static PyObject *type##_get_clink_ptr(PyObject *self) \
|
|
{ \
|
|
return PyLong_FromUnsignedLongLong( \
|
|
(unsigned PY_LONG_LONG)pyobj_get_clink(self)); \
|
|
}
|
|
#else
|
|
// SWIG does not expand macros and thus those definitions won't be wrapped
|
|
// Use DECLARE_PY_CLINKED_OBJECT(type) inside the .i file
|
|
#define DECLARE_PY_CLINKED_OBJECT(type)
|
|
#endif // __PYWRAPS__
|
|
|
|
//---------------------------------------------------------------------------
|
|
class CGilStateAuto
|
|
{
|
|
private:
|
|
PyGILState_STATE state;
|
|
public:
|
|
CGilStateAuto()
|
|
{
|
|
state = PyGILState_Ensure();
|
|
}
|
|
|
|
~CGilStateAuto()
|
|
{
|
|
PyGILState_Release(state);
|
|
}
|
|
};
|
|
// Declare a variable to acquire/release the GIL
|
|
#define PYW_GIL_AUTO_ENSURE CGilStateAuto GIL_STATE_AUTO
|
|
|
|
// Macros to acquire/release GIL in a given scope
|
|
#define PYW_GIL_ENSURE_N(name) PyGILState_STATE gil_state##name = PyGILState_Ensure()
|
|
#define PYW_GIL_RELEASE_N(name) PyGILState_Release(gil_state##name)
|
|
|
|
#define PYW_GIL_ENSURE PYW_GIL_ENSURE_N(_)
|
|
#define PYW_GIL_RELEASE PYW_GIL_RELEASE_N(_)
|
|
|
|
//------------------------------------------------------------------------
|
|
// All the exported functions from PyWraps are forward declared here
|
|
insn_t *insn_t_get_clink(PyObject *self);
|
|
op_t *op_t_get_clink(PyObject *self);
|
|
|
|
// Returns a reference to a class
|
|
PyObject *get_idaapi_attr(const char *attr);
|
|
|
|
// Returns a reference to a class by its ID
|
|
PyObject *get_idaapi_attr_by_id(const int class_id);
|
|
|
|
// Tries to import a module and swallows the exception if it fails and returns NULL
|
|
PyObject *PyW_TryImportModule(const char *name);
|
|
|
|
// Tries to get an attribute and swallows the exception if it fails and returns NULL
|
|
PyObject *PyW_TryGetAttrString(PyObject *py_var, const char *attr);
|
|
|
|
// Returns the linked object (void *) from a PyObject
|
|
void *pyobj_get_clink(PyObject *pyobj);
|
|
|
|
// Converts a Python number (LONGLONG or normal integer) to an IDC variable (VT_LONG or VT_INT64)
|
|
bool PyW_GetNumberAsIDC(PyObject *py_var, idc_value_t *idc_var);
|
|
|
|
// Returns a qstring from a Python attribute string
|
|
bool PyW_GetStringAttr(
|
|
PyObject *py_obj,
|
|
const char *attr_name,
|
|
qstring *str);
|
|
|
|
// Converts a Python number to an uint64 and indicates whether the number was a long number
|
|
bool PyW_GetNumber(PyObject *py_var, uint64 *num, bool *is_64 = NULL);
|
|
|
|
// Checks if an Python object can be treated like a sequence
|
|
bool PyW_IsSequenceType(PyObject *obj);
|
|
|
|
// Returns an error string from the last exception (and clears it)
|
|
bool PyW_GetError(qstring *out = NULL, bool clear_err = true);
|
|
bool PyW_GetError(char *buf, size_t bufsz, bool clear_err = true);
|
|
|
|
// If an error occured (it calls PyGetError) it displays it and return TRUE
|
|
// This function is used when calling callbacks
|
|
bool PyW_ShowCbErr(const char *cb_name);
|
|
|
|
// Utility function to create a class instance whose constructor takes zero arguments
|
|
PyObject *create_idaapi_class_instance0(const char *clsname);
|
|
|
|
// Utility function to create linked class instances
|
|
PyObject *create_idaapi_linked_class_instance(const char *clsname, void *lnk);
|
|
|
|
// Returns the string representation of a PyObject
|
|
bool PyW_ObjectToString(PyObject *obj, qstring *out);
|
|
|
|
// Utility function to convert a python object to an IDC object
|
|
// and sets a python exception on failure.
|
|
bool convert_pyobj_to_idc_exc(PyObject *py_obj, idc_value_t *idc_obj);
|
|
|
|
// Creates and initializes an IDC exception
|
|
error_t PyW_CreateIdcException(idc_value_t *res, const char *msg);
|
|
|
|
//
|
|
// Conversion functions
|
|
//
|
|
bool pyw_convert_idc_args(
|
|
const idc_value_t args[],
|
|
int nargs,
|
|
ppyobject_vec_t &pargs,
|
|
boolvec_t *decref,
|
|
char *errbuf = NULL,
|
|
size_t errbufsize = 0);
|
|
|
|
void pyw_free_idc_args(
|
|
ppyobject_vec_t &pargs,
|
|
boolvec_t *decref = NULL);
|
|
|
|
// Converts Python variable to IDC variable
|
|
// gvar_sn is used in case the Python object was a created from a call to idcvar_to_pyvar and the IDC object was a VT_REF
|
|
int pyvar_to_idcvar(
|
|
PyObject *py_var,
|
|
idc_value_t *idc_var,
|
|
int *gvar_sn = NULL);
|
|
|
|
// Converts from IDC to Python
|
|
// We support converting VT_REF IDC variable types
|
|
int idcvar_to_pyvar(
|
|
const idc_value_t &idc_var,
|
|
PyObject **py_var);
|
|
|
|
// Walks a Python list or Sequence and calls the callback
|
|
Py_ssize_t pyvar_walk_list(
|
|
PyObject *py_list,
|
|
int (idaapi *cb)(PyObject *py_item, Py_ssize_t index, void *ud) = NULL,
|
|
void *ud = NULL);
|
|
|
|
// Converts an intvec_t to a Python list object
|
|
PyObject *PyW_IntVecToPyList(const intvec_t &intvec);
|
|
|
|
// Converts an Python list to an intvec
|
|
bool PyW_PyListToIntVec(PyObject *py_list, intvec_t &intvec);
|
|
|
|
// Converts a Python list to a qstrvec
|
|
bool PyW_PyListToStrVec(PyObject *py_list, qstrvec_t &strvec);
|
|
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
// notify_when()
|
|
//
|
|
bool pywraps_nw_term();
|
|
bool pywraps_nw_notify(int slot, ...);
|
|
bool pywraps_nw_init();
|
|
|
|
//---------------------------------------------------------------------------
|
|
bool pywraps_check_autoscripts(char *buf, size_t bufsize);
|
|
|
|
// [De]Initializes PyWraps
|
|
bool init_pywraps();
|
|
void deinit_pywraps();
|
|
|
|
#endif |