2010-11-10 13:58:08 +00:00
|
|
|
%ignore cancellable_graph_t;
|
|
|
|
%ignore gdl_graph_t;
|
2010-06-28 12:36:40 +00:00
|
|
|
|
2010-11-10 13:58:08 +00:00
|
|
|
%ignore intmap_t;
|
|
|
|
%ignore intset_t;
|
|
|
|
%ignore intseq_t;
|
|
|
|
%ignore node_set_t;
|
2010-06-28 12:36:40 +00:00
|
|
|
%ignore qflow_chart_t::blocks;
|
|
|
|
%ignore flow_chart_t;
|
|
|
|
%ignore default_graph_format;
|
|
|
|
%ignore setup_graph_subsystem;
|
|
|
|
%ignore qbasic_block_t::succ;
|
|
|
|
%ignore qbasic_block_t::pred;
|
|
|
|
|
|
|
|
%include "gdl.hpp"
|
|
|
|
|
|
|
|
%extend qflow_chart_t
|
|
|
|
{
|
|
|
|
qbasic_block_t *__getitem__(int n)
|
|
|
|
{
|
|
|
|
return &(self->blocks[n]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
%pythoncode %{
|
|
|
|
#<pycode(py_gdl)>
|
|
|
|
# -----------------------------------------------------------------------
|
2010-12-17 16:50:35 +00:00
|
|
|
class BasicBlock(object):
|
2010-06-28 12:36:40 +00:00
|
|
|
"""Basic block class. It is returned by the Flowchart class"""
|
2010-12-17 16:50:35 +00:00
|
|
|
def __init__(self, id, bb, fc):
|
|
|
|
self._fc = fc
|
|
|
|
|
2010-06-28 12:36:40 +00:00
|
|
|
self.id = id
|
|
|
|
"""Basic block ID"""
|
2010-12-17 16:50:35 +00:00
|
|
|
|
2010-06-28 12:36:40 +00:00
|
|
|
self.startEA = bb.startEA
|
|
|
|
"""startEA of basic block"""
|
2010-12-17 16:50:35 +00:00
|
|
|
|
2010-06-28 12:36:40 +00:00
|
|
|
self.endEA = bb.endEA
|
|
|
|
"""endEA of basic block"""
|
2010-12-17 16:50:35 +00:00
|
|
|
|
|
|
|
self.type = self._fc._q.calc_block_type(self.id)
|
2010-06-28 12:36:40 +00:00
|
|
|
"""Block type (check fc_block_type_t enum)"""
|
|
|
|
|
2010-12-17 16:50:35 +00:00
|
|
|
|
2010-06-28 12:36:40 +00:00
|
|
|
def preds(self):
|
|
|
|
"""
|
|
|
|
Iterates the predecessors list
|
|
|
|
"""
|
2010-12-17 16:50:35 +00:00
|
|
|
q = self._fc._q
|
|
|
|
for i in xrange(0, self._fc._q.npred(self.id)):
|
|
|
|
yield self._fc[q.pred(self.id, i)]
|
2010-06-28 12:36:40 +00:00
|
|
|
|
2010-12-17 16:50:35 +00:00
|
|
|
|
2010-06-28 12:36:40 +00:00
|
|
|
def succs(self):
|
|
|
|
"""
|
|
|
|
Iterates the successors list
|
|
|
|
"""
|
2010-12-17 16:50:35 +00:00
|
|
|
q = self._fc._q
|
2010-06-28 12:36:40 +00:00
|
|
|
for i in xrange(0, q.nsucc(self.id)):
|
2010-12-17 16:50:35 +00:00
|
|
|
yield self._fc[q.succ(self.id, i)]
|
2010-06-28 12:36:40 +00:00
|
|
|
|
|
|
|
# -----------------------------------------------------------------------
|
2010-12-17 16:50:35 +00:00
|
|
|
class FlowChart(object):
|
2010-06-28 12:36:40 +00:00
|
|
|
"""
|
|
|
|
Flowchart class used to determine basic blocks.
|
|
|
|
Check ex_gdl_qflow_chart.py for sample usage.
|
|
|
|
"""
|
|
|
|
def __init__(self, f=None, bounds=None, flags=0):
|
|
|
|
"""
|
|
|
|
Constructor
|
|
|
|
@param f: A func_t type, use get_func(ea) to get a reference
|
|
|
|
@param bounds: A tuple of the form (start, end). Used if "f" is None
|
|
|
|
@param flags: one of the FC_xxxx flags. One interesting flag is FC_PREDS
|
|
|
|
"""
|
2010-12-17 16:50:35 +00:00
|
|
|
if (f is None) and (bounds is None or type(bounds) != types.TupleType):
|
2010-06-28 12:36:40 +00:00
|
|
|
raise Exception("Please specifiy either a function or start/end pair")
|
2010-12-17 16:50:35 +00:00
|
|
|
|
|
|
|
if bounds is None:
|
2010-06-28 12:36:40 +00:00
|
|
|
bounds = (BADADDR, BADADDR)
|
2010-12-17 16:50:35 +00:00
|
|
|
|
|
|
|
# Create the flowchart
|
2010-06-28 12:36:40 +00:00
|
|
|
self._q = qflow_chart_t("", f, bounds[0], bounds[1], flags)
|
2010-12-17 16:50:35 +00:00
|
|
|
|
|
|
|
size = property(lambda self: self._q.size())
|
|
|
|
"""Number of blocks in the flow chart"""
|
|
|
|
|
2010-06-28 12:36:40 +00:00
|
|
|
|
|
|
|
def refresh():
|
2010-12-17 16:50:35 +00:00
|
|
|
"""Refreshes the flow chart"""
|
2010-06-28 12:36:40 +00:00
|
|
|
self._q.refresh()
|
2010-12-17 16:50:35 +00:00
|
|
|
|
2010-06-28 12:36:40 +00:00
|
|
|
|
2011-04-18 16:07:00 +00:00
|
|
|
def _getitem(self, index):
|
|
|
|
return BasicBlock(index, self._q[index], self)
|
|
|
|
|
|
|
|
|
|
|
|
def __iter__(self):
|
|
|
|
return (self._getitem(index) for index in xrange(0, self.size))
|
|
|
|
|
|
|
|
|
2010-06-28 12:36:40 +00:00
|
|
|
def __getitem__(self, index):
|
|
|
|
"""
|
|
|
|
Returns a basic block
|
2010-12-17 16:50:35 +00:00
|
|
|
|
2010-06-28 12:36:40 +00:00
|
|
|
@return: BasicBlock
|
|
|
|
"""
|
|
|
|
if index >= self.size:
|
2011-04-18 16:07:00 +00:00
|
|
|
raise KeyError
|
2010-12-17 16:50:35 +00:00
|
|
|
else:
|
2011-04-18 16:07:00 +00:00
|
|
|
return self._getitem(index)
|
2010-12-17 16:50:35 +00:00
|
|
|
|
2010-06-28 12:36:40 +00:00
|
|
|
#</pycode(py_gdl)>
|
|
|
|
%}
|