mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-24 23:06:36 +00:00
2865 lines
74 KiB
C
2865 lines
74 KiB
C
/* Copyright radare2 - 2014-2015 - pancake, ret2libc */
|
|
|
|
#include <r_core.h>
|
|
#include <r_cons.h>
|
|
#include <ctype.h>
|
|
#include <limits.h>
|
|
|
|
static const char *mousemodes[] = { "canvas-y", "canvas-x", "node-y", "node-x", NULL };
|
|
static int mousemode = 0;
|
|
|
|
#define BORDER 3
|
|
#define BORDER_WIDTH 4
|
|
#define BORDER_HEIGHT 3
|
|
#define MARGIN_TEXT_X 2
|
|
#define MARGIN_TEXT_Y 2
|
|
#define HORIZONTAL_NODE_SPACING 6
|
|
#define VERTICAL_NODE_SPACING 4
|
|
#define MIN_NODE_WIDTH 18
|
|
#define MIN_NODE_HEIGTH BORDER_HEIGHT
|
|
#define TITLE_LEN 128
|
|
#define DEFAULT_SPEED 1
|
|
#define PAGEKEY_SPEED 15
|
|
#define SMALLNODE_TEXT_CUR "<@@@@@@>"
|
|
#define SMALLNODE_MIN_WIDTH 8
|
|
#define SMALLNODE_TITLE_LEN 4
|
|
#define SMALLNODE_CENTER_X 3
|
|
|
|
#define ZOOM_STEP 10
|
|
#define ZOOM_DEFAULT 100
|
|
|
|
#define hash_set(sdb,k,v) (sdb_num_set (sdb, sdb_fmt (0, "%"PFMT64u, (ut64)(size_t)k), (ut64)(size_t)v, 0))
|
|
#define hash_get(sdb,k) (sdb_num_get (sdb, sdb_fmt (0, "%"PFMT64u, (ut64)(size_t)k), NULL))
|
|
#define hash_get_rnode(sdb,k) ((RGraphNode *)(size_t)hash_get (sdb, k))
|
|
#define hash_get_rlist(sdb,k) ((RList *)(size_t)hash_get (sdb, k))
|
|
#define hash_get_int(sdb,k) ((int)hash_get (sdb, k))
|
|
|
|
#define get_anode(gn) (gn ? (RANode *)gn->data : NULL)
|
|
|
|
#define graph_foreach_anode(list, it, pos, anode) \
|
|
if (list) for (it = list->head; it && (pos = it->data) && (pos) && (anode = (RANode *)pos->data); it = it->n)
|
|
|
|
struct len_pos_t {
|
|
int len;
|
|
int pos;
|
|
};
|
|
|
|
struct dist_t {
|
|
const RGraphNode *from;
|
|
const RGraphNode *to;
|
|
int dist;
|
|
};
|
|
|
|
struct g_cb {
|
|
RAGraph *graph;
|
|
RANodeCallback node_cb;
|
|
RAEdgeCallback edge_cb;
|
|
void *data;
|
|
};
|
|
|
|
typedef struct ascii_edge_t {
|
|
RANode *from;
|
|
RANode *to;
|
|
RList *x, *y;
|
|
int is_reversed;
|
|
} AEdge;
|
|
|
|
struct layer_t {
|
|
int n_nodes;
|
|
RGraphNode **nodes;
|
|
int position;
|
|
int height;
|
|
};
|
|
|
|
struct agraph_refresh_data {
|
|
RCore *core;
|
|
RAGraph *g;
|
|
RAnalFunction **fcn;
|
|
int fs;
|
|
};
|
|
|
|
#define G(x,y) r_cons_canvas_gotoxy (g->can, x, y)
|
|
#define W(x) r_cons_canvas_write (g->can, x)
|
|
#define B(x,y,w,h) r_cons_canvas_box(g->can, x,y,w,h, g->color_box)
|
|
#define B1(x,y,w,h) r_cons_canvas_box(g->can, x,y,w,h, g->color_box2)
|
|
#define B2(x,y,w,h) r_cons_canvas_box(g->can, x,y,w,h, g->color_box3)
|
|
#define F(x,y,x2,y2,c) r_cons_canvas_fill(g->can, x,y,x2,y2,c,0)
|
|
|
|
static bool is_offset(const RAGraph *g) {
|
|
return g->mode == R_AGRAPH_MODE_OFFSET || g->mode == R_AGRAPH_MODE_ESIL_OFFSET;
|
|
}
|
|
static bool is_mini(const RAGraph *g) {
|
|
return g->mode == R_AGRAPH_MODE_MINI;
|
|
}
|
|
static bool is_esil(const RAGraph *g) {
|
|
return g->mode == R_AGRAPH_MODE_ESIL || g->mode == R_AGRAPH_MODE_ESIL_OFFSET;
|
|
}
|
|
static int next_mode(int mode) {
|
|
return (mode + 1) % R_AGRAPH_MODE_MAX;
|
|
}
|
|
static int prev_mode(int mode) {
|
|
return (mode + R_AGRAPH_MODE_MAX - 1) % R_AGRAPH_MODE_MAX;
|
|
}
|
|
static const char *mode2str(const RAGraph *g, const char *prefix) {
|
|
static char m[20];
|
|
const char *submode;
|
|
|
|
if (is_mini (g)) submode = "SMALL";
|
|
else if (is_esil (g) && is_offset (g)) submode = "ESIL-OFF";
|
|
else if (is_esil (g)) submode = "ESIL";
|
|
else if (is_offset (g)) submode = "OFF";
|
|
else submode = "NORM";
|
|
|
|
snprintf(m, sizeof (m), "%s-%s", prefix, submode);
|
|
return m;
|
|
}
|
|
|
|
static char *get_title (ut64 addr) {
|
|
return r_str_newf ("0x%"PFMT64x, addr);
|
|
}
|
|
|
|
static void update_node_dimension(const RGraph *g, int is_small, int zoom) {
|
|
const RList *nodes = r_graph_get_nodes (g);
|
|
RGraphNode *gn;
|
|
RListIter *it;
|
|
RANode *n;
|
|
|
|
graph_foreach_anode (nodes, it, gn, n) {
|
|
if (is_small) {
|
|
n->h = 1;
|
|
n->w = SMALLNODE_MIN_WIDTH;
|
|
} else {
|
|
unsigned int len;
|
|
|
|
n->w = r_str_bounds (n->body, (int *)&n->h);
|
|
len = strlen (n->title) + MARGIN_TEXT_X;
|
|
if (len > INT_MAX) len = INT_MAX;
|
|
n->w = R_MAX (n->w, (int)len);
|
|
n->w += BORDER_WIDTH;
|
|
n->h += BORDER_HEIGHT;
|
|
/* scale node by zoom */
|
|
n->w = R_MAX (MIN_NODE_WIDTH, (n->w * zoom) / 100);
|
|
n->h = R_MAX (MIN_NODE_HEIGTH, (n->h * zoom) / 100);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void small_RANode_print(const RAGraph *g, const RANode *n, int cur) {
|
|
char title[TITLE_LEN];
|
|
int x, delta_x = 0;
|
|
|
|
if (!G (n->x + SMALLNODE_CENTER_X, n->y) &&
|
|
!G (n->x + SMALLNODE_CENTER_X + n->w, n->y)) return;
|
|
|
|
x = n->x + SMALLNODE_CENTER_X + g->can->sx;
|
|
if (x < 0) {
|
|
delta_x = -x;
|
|
}
|
|
G (n->x + SMALLNODE_CENTER_X + delta_x, n->y);
|
|
|
|
if (cur) {
|
|
W(&SMALLNODE_TEXT_CUR[delta_x]);
|
|
(void)G (-g->can->sx, -g->can->sy + 2);
|
|
snprintf (title, sizeof (title) - 1,
|
|
"%s:", n->title);
|
|
W (title);
|
|
(void)G (-g->can->sx, -g->can->sy + 3);
|
|
W (n->body);
|
|
} else {
|
|
char *str = "____";
|
|
if (n->title) {
|
|
int l = strlen (n->title);
|
|
|
|
str = n->title;
|
|
if (l > SMALLNODE_TITLE_LEN) {
|
|
str += l - SMALLNODE_TITLE_LEN;
|
|
}
|
|
}
|
|
snprintf (title, sizeof (title) - 1, "[_%s_]", str);
|
|
W(title + delta_x);
|
|
}
|
|
return;
|
|
}
|
|
|
|
static void normal_RANode_print(const RAGraph *g, const RANode *n, int cur) {
|
|
unsigned int center_x = 0, center_y = 0;
|
|
unsigned int delta_x = 0, delta_txt_x = 0;
|
|
unsigned int delta_y = 0, delta_txt_y = 0;
|
|
char title[TITLE_LEN];
|
|
char *body;
|
|
int x, y;
|
|
|
|
x = n->x + g->can->sx;
|
|
y = n->y + g->can->sy;
|
|
if (x + MARGIN_TEXT_X < 0)
|
|
delta_x = -(x + MARGIN_TEXT_X);
|
|
if (x + n->w < -MARGIN_TEXT_X)
|
|
return;
|
|
if (y < -1)
|
|
delta_y = R_MIN (n->h - BORDER_HEIGHT - 1, -y - MARGIN_TEXT_Y);
|
|
|
|
/* print the title */
|
|
if (cur) {
|
|
snprintf (title, sizeof (title)-1,
|
|
"[%s]", n->title);
|
|
} else {
|
|
snprintf (title, sizeof (title)-1,
|
|
" %s ", n->title);
|
|
}
|
|
if (delta_x < strlen(title) && G(n->x + MARGIN_TEXT_X + delta_x, n->y + 1))
|
|
W(title + delta_x);
|
|
|
|
/* print the body */
|
|
if (g->zoom > ZOOM_DEFAULT) {
|
|
center_x = (g->zoom - ZOOM_DEFAULT) / 20;
|
|
center_y = (g->zoom - ZOOM_DEFAULT) / 30;
|
|
delta_txt_x = R_MIN (delta_x, center_x);
|
|
delta_txt_y = R_MIN (delta_y, center_y);
|
|
}
|
|
|
|
if (G(n->x + MARGIN_TEXT_X + delta_x + center_x - delta_txt_x,
|
|
n->y + MARGIN_TEXT_Y + delta_y + center_y - delta_txt_y)) {
|
|
unsigned int body_x = center_x >= delta_x ? 0 : delta_x - center_x;
|
|
unsigned int body_y = center_y >= delta_y ? 0 : delta_y - center_y;
|
|
unsigned int body_h = BORDER_HEIGHT >= n->h ? 1 : n->h - BORDER_HEIGHT;
|
|
|
|
if (g->zoom < ZOOM_DEFAULT) body_h--;
|
|
if (body_y + 1 <= body_h) {
|
|
body = r_str_ansi_crop (n->body,
|
|
body_x, body_y,
|
|
n->w - BORDER_WIDTH,
|
|
body_h);
|
|
if (body) {
|
|
W (body);
|
|
if (g->zoom < ZOOM_DEFAULT) W ("\n");
|
|
free (body);
|
|
} else {
|
|
W (n->body);
|
|
}
|
|
}
|
|
/* print some dots when the body is cropped because of zoom */
|
|
if (body_y <= body_h && g->zoom < ZOOM_DEFAULT) {
|
|
char *dots = "...";
|
|
if (delta_x < strlen(dots)) {
|
|
dots += delta_x;
|
|
W (dots);
|
|
}
|
|
}
|
|
}
|
|
|
|
// TODO: check if node is traced or not and hsow proper color
|
|
// This info must be stored inside RANode* from RCore*
|
|
if (cur) {
|
|
B1 (n->x, n->y, n->w, n->h);
|
|
} else {
|
|
B (n->x, n->y, n->w, n->h);
|
|
}
|
|
}
|
|
|
|
static int **get_crossing_matrix (const RGraph *g,
|
|
const struct layer_t layers[],
|
|
int maxlayer, int i, int from_up,
|
|
int *n_rows) {
|
|
int len = layers[i].n_nodes;
|
|
int **m;
|
|
int j;
|
|
|
|
m = R_NEWS0 (int *, len);
|
|
if (!m)
|
|
return NULL;
|
|
|
|
for (j = 0; j < len; ++j) {
|
|
m[j] = R_NEWS0 (int, len);
|
|
if (!m[j])
|
|
goto err_row;
|
|
}
|
|
|
|
/* calculate crossings between layer i and layer i-1 */
|
|
/* consider the crossings generated by each pair of edges */
|
|
if (i > 0 && from_up) {
|
|
for (j = 0; j < layers[i - 1].n_nodes; ++j) {
|
|
const RGraphNode *gj = layers[i - 1].nodes[j];
|
|
const RList *neigh = r_graph_get_neighbours (g, gj);
|
|
RGraphNode *gk;
|
|
RListIter *itk;
|
|
|
|
r_list_foreach (neigh, itk, gk) {
|
|
int s;
|
|
for (s = 0; s < j; ++s) {
|
|
const RGraphNode *gs = layers[i - 1].nodes[s];
|
|
const RList *neigh_s = r_graph_get_neighbours (g, gs);
|
|
RGraphNode *gt;
|
|
RListIter *itt;
|
|
|
|
r_list_foreach (neigh_s, itt, gt) {
|
|
const RANode *ak, *at; /* k and t should be "indexes" on layer i */
|
|
|
|
if (gt == gk) continue;
|
|
ak = get_anode (gk);
|
|
at = get_anode (gt);
|
|
if (ak->layer != i || at->layer != i) {
|
|
/* eprintf("\"%s\" (%d) or \"%s\" (%d) are not on the right layer (%d)\n", */
|
|
/* ak->title, ak->layer, */
|
|
/* at->title, at->layer, */
|
|
/* i); */
|
|
|
|
continue;
|
|
}
|
|
m[ak->pos_in_layer][at->pos_in_layer]++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* calculate crossings between layer i and layer i+1 */
|
|
if (i < maxlayer - 1 && !from_up) {
|
|
for (j = 0; j < layers[i].n_nodes; ++j) {
|
|
const RGraphNode *gj = layers[i].nodes[j];
|
|
const RList *neigh = r_graph_get_neighbours (g, gj);
|
|
const RANode *ak, *aj = get_anode (gj);
|
|
RGraphNode *gk;
|
|
RListIter *itk;
|
|
|
|
graph_foreach_anode (neigh, itk, gk, ak) {
|
|
int s;
|
|
for (s = 0; s < layers[i].n_nodes; ++s) {
|
|
const RGraphNode *gs = layers[i].nodes[s];
|
|
const RList *neigh_s;
|
|
RGraphNode *gt;
|
|
RListIter *itt;
|
|
const RANode *at, *as = get_anode (gs);
|
|
|
|
if (gs == gj) continue;
|
|
neigh_s = r_graph_get_neighbours (g, gs);
|
|
graph_foreach_anode (neigh_s, itt, gt, at) {
|
|
if (at->pos_in_layer < ak->pos_in_layer)
|
|
m[aj->pos_in_layer][as->pos_in_layer]++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (n_rows)
|
|
*n_rows = len;
|
|
return m;
|
|
|
|
err_row:
|
|
for (i = 0; i < len; ++i) {
|
|
if (m[i])
|
|
free (m[i]);
|
|
}
|
|
free (m);
|
|
return NULL;
|
|
}
|
|
|
|
static int layer_sweep (const RGraph *g, const struct layer_t layers[],
|
|
int maxlayer, int i, int from_up) {
|
|
int **cross_matrix;
|
|
RGraphNode *u, *v;
|
|
const RANode *au, *av;
|
|
int n_rows, j, changed = false;
|
|
int len = layers[i].n_nodes;
|
|
|
|
cross_matrix = get_crossing_matrix (g, layers, maxlayer, i, from_up, &n_rows);
|
|
if (!cross_matrix) return false;
|
|
|
|
for (j = 0; j < len - 1; ++j) {
|
|
int auidx, avidx;
|
|
|
|
u = layers[i].nodes[j];
|
|
v = layers[i].nodes[j + 1];
|
|
au = get_anode (u);
|
|
av = get_anode (v);
|
|
auidx = au->pos_in_layer;
|
|
avidx = av->pos_in_layer;
|
|
|
|
if (cross_matrix[auidx][avidx] > cross_matrix[avidx][auidx]) {
|
|
/* swap elements */
|
|
layers[i].nodes[j] = v;
|
|
layers[i].nodes[j + 1] = u;
|
|
changed = true;
|
|
}
|
|
}
|
|
|
|
/* update position in the layer of each node. During the swap of some
|
|
* elements we didn't swap also the pos_in_layer because the cross_matrix
|
|
* is indexed by it, so do it now! */
|
|
for (j = 0; j < layers[i].n_nodes; ++j) {
|
|
RANode *n = get_anode (layers[i].nodes[j]);
|
|
n->pos_in_layer = j;
|
|
}
|
|
|
|
for (j = 0; j < n_rows; ++j)
|
|
free (cross_matrix[j]);
|
|
free (cross_matrix);
|
|
return changed;
|
|
}
|
|
|
|
static void view_cyclic_edge (const RGraphEdge *e, const RGraphVisitor *vis) {
|
|
const RAGraph *g = (RAGraph *)vis->data;
|
|
RGraphEdge *new_e = R_NEW (RGraphEdge);
|
|
|
|
new_e->from = e->from;
|
|
new_e->to = e->to;
|
|
new_e->nth = e->nth;
|
|
r_list_append (g->back_edges, new_e);
|
|
}
|
|
|
|
static int get_depth (Sdb *path, const RGraphNode *n) {
|
|
int res = 0;
|
|
while ((n = hash_get_rnode (path, n)) != NULL) {
|
|
res++;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
static void set_layer (const RGraphEdge *e, const RGraphVisitor *vis) {
|
|
Sdb *path = (Sdb *)vis->data;
|
|
int bdepth, adepth;
|
|
|
|
adepth = get_depth (path, e->from);
|
|
bdepth = get_depth (path, e->to);
|
|
|
|
if (adepth + 1 > bdepth)
|
|
hash_set (path, e->to, e->from);
|
|
}
|
|
|
|
static void view_dummy (const RGraphEdge *e, const RGraphVisitor *vis) {
|
|
const RANode *a = get_anode (e->from);
|
|
const RANode *b = get_anode (e->to);
|
|
RList *long_edges = (RList *)vis->data;
|
|
|
|
if (R_ABS (a->layer - b->layer) > 1) {
|
|
RGraphEdge *new_e = R_NEW (RGraphEdge);
|
|
new_e->from = e->from;
|
|
new_e->to = e->to;
|
|
new_e->nth = e->nth;
|
|
r_list_append (long_edges, new_e);
|
|
}
|
|
}
|
|
|
|
/* find a set of edges that, removed, makes the graph acyclic */
|
|
/* invert the edges identified in the previous step */
|
|
static void remove_cycles (RAGraph *g) {
|
|
RGraphVisitor cyclic_vis = { NULL, NULL, NULL, NULL, NULL, NULL };
|
|
const RGraphEdge *e;
|
|
const RListIter *it;
|
|
|
|
g->back_edges = r_list_new();
|
|
cyclic_vis.back_edge = (RGraphEdgeCallback)view_cyclic_edge;
|
|
cyclic_vis.data = g;
|
|
r_graph_dfs (g->graph, &cyclic_vis);
|
|
|
|
r_list_foreach (g->back_edges, it, e) {
|
|
RANode *from, *to;
|
|
from = e->from ? get_anode (e->from) : NULL;
|
|
to = e->to ? get_anode (e->to) : NULL;
|
|
r_agraph_del_edge (g, from, to);
|
|
r_agraph_add_edge_at (g, to, from, e->nth);
|
|
}
|
|
}
|
|
|
|
/* assign a layer to each node of the graph */
|
|
static void assign_layers (const RAGraph *g) {
|
|
RGraphVisitor layer_vis = { NULL, NULL, NULL, NULL, NULL, NULL };
|
|
Sdb *path_layers = sdb_new0 ();
|
|
const RGraphNode *gn;
|
|
const RListIter *it;
|
|
RANode *n;
|
|
|
|
layer_vis.data = path_layers;
|
|
layer_vis.tree_edge = (RGraphEdgeCallback)set_layer;
|
|
layer_vis.fcross_edge = (RGraphEdgeCallback)set_layer;
|
|
r_graph_dfs (g->graph, &layer_vis);
|
|
|
|
graph_foreach_anode (r_graph_get_nodes (g->graph), it, gn, n) {
|
|
n->layer = get_depth (path_layers, gn);
|
|
}
|
|
|
|
sdb_free (path_layers);
|
|
}
|
|
|
|
static int find_edge (const RGraphEdge *a, const RGraphEdge *b) {
|
|
return a->from == b->to && a->to == b->from ? 0 : 1;
|
|
}
|
|
|
|
static int is_reversed (const RAGraph *g, const RGraphEdge *e) {
|
|
return r_list_find (g->back_edges, e, (RListComparator)find_edge) ? true : false;
|
|
}
|
|
|
|
/* add dummy nodes when there are edges that span multiple layers */
|
|
static void create_dummy_nodes (RAGraph *g) {
|
|
RGraphVisitor dummy_vis = { NULL, NULL, NULL, NULL, NULL, NULL };
|
|
const RListIter *it;
|
|
const RGraphEdge *e;
|
|
|
|
g->long_edges = r_list_new ();
|
|
dummy_vis.data = g->long_edges;
|
|
dummy_vis.tree_edge = (RGraphEdgeCallback)view_dummy;
|
|
dummy_vis.fcross_edge = (RGraphEdgeCallback)view_dummy;
|
|
r_graph_dfs (g->graph, &dummy_vis);
|
|
|
|
r_list_foreach (g->long_edges, it, e) {
|
|
RANode *from = get_anode (e->from);
|
|
RANode *to = get_anode (e->to);
|
|
int diff_layer = R_ABS (from->layer - to->layer);
|
|
RANode *prev = get_anode(e->from);
|
|
int i, nth = e->nth;
|
|
|
|
r_agraph_del_edge (g, from, to);
|
|
for (i = 1; i < diff_layer; ++i) {
|
|
RANode *dummy = r_agraph_add_node (g, NULL, NULL);
|
|
if (!dummy) return;
|
|
dummy->is_dummy = true;
|
|
dummy->layer = from->layer + i;
|
|
dummy->is_reversed = is_reversed (g, e);
|
|
dummy->w = 1;
|
|
r_agraph_add_edge_at (g, prev, dummy, nth);
|
|
|
|
prev = dummy;
|
|
nth = -1;
|
|
}
|
|
r_graph_add_edge (g->graph, prev->gnode, e->to);
|
|
}
|
|
}
|
|
|
|
/* create layers and assign an initial ordering of the nodes into them */
|
|
static void create_layers (RAGraph *g) {
|
|
const RList *nodes = r_graph_get_nodes (g->graph);
|
|
RGraphNode *gn;
|
|
const RListIter *it;
|
|
RANode *n;
|
|
int i;
|
|
|
|
/* identify max layer */
|
|
g->n_layers = 0;
|
|
graph_foreach_anode (nodes, it, gn, n) {
|
|
if (n->layer > g->n_layers)
|
|
g->n_layers = n->layer;
|
|
}
|
|
|
|
/* create a starting ordering of nodes for each layer */
|
|
g->n_layers++;
|
|
if (sizeof (struct layer_t) * g->n_layers < g->n_layers) return;
|
|
g->layers = R_NEWS0 (struct layer_t, g->n_layers);
|
|
|
|
graph_foreach_anode (nodes, it, gn, n)
|
|
g->layers[n->layer].n_nodes++;
|
|
|
|
for (i = 0; i < g->n_layers; ++i) {
|
|
if (sizeof(RGraphNode*) * g->layers[i].n_nodes < g->layers[i].n_nodes) {
|
|
//FIXME how to handle properly this error ret2libc?
|
|
continue;
|
|
}
|
|
g->layers[i].nodes = R_NEWS0 (RGraphNode *,
|
|
1 + g->layers[i].n_nodes);
|
|
g->layers[i].position = 0;
|
|
}
|
|
graph_foreach_anode (nodes, it, gn, n) {
|
|
n->pos_in_layer = g->layers[n->layer].position;
|
|
g->layers[n->layer].nodes[g->layers[n->layer].position++] = gn;
|
|
}
|
|
}
|
|
|
|
/* layer-by-layer sweep */
|
|
/* it permutes each layer, trying to find the best ordering for each layer
|
|
* to minimize the number of crossing edges */
|
|
static void minimize_crossings (const RAGraph *g) {
|
|
int i, cross_changed, max_changes = 4096;
|
|
|
|
do {
|
|
cross_changed = false;
|
|
--max_changes;
|
|
|
|
for (i = 0; i < g->n_layers; ++i)
|
|
cross_changed |= layer_sweep (g->graph, g->layers, g->n_layers, i, true);
|
|
} while (cross_changed && max_changes);
|
|
|
|
max_changes = 4096;
|
|
|
|
do {
|
|
cross_changed = false;
|
|
--max_changes;
|
|
|
|
for (i = g->n_layers - 1; i >= 0; --i)
|
|
cross_changed |= layer_sweep (g->graph, g->layers, g->n_layers, i, false);
|
|
} while (cross_changed && max_changes);
|
|
}
|
|
|
|
static int find_dist (const struct dist_t *a, const struct dist_t *b) {
|
|
return a->from == b->from && a->to == b->to ? 0 : 1;
|
|
}
|
|
|
|
/* returns the distance between two nodes */
|
|
/* if the distance between two nodes were explicitly set, returns that;
|
|
* otherwise calculate the distance of two nodes on the same layer */
|
|
static int dist_nodes (const RAGraph *g, const RGraphNode *a, const RGraphNode *b) {
|
|
struct dist_t d;
|
|
const RANode *aa, *ab;
|
|
RListIter *it;
|
|
int res = 0;
|
|
|
|
if (g->dists) {
|
|
d.from = a;
|
|
d.to = b;
|
|
it = r_list_find (g->dists, &d, (RListComparator)find_dist);
|
|
if (it) {
|
|
struct dist_t *old = (struct dist_t *)r_list_iter_get_data (it);
|
|
return old->dist;
|
|
}
|
|
}
|
|
|
|
aa = get_anode (a);
|
|
ab = get_anode (b);
|
|
if (aa && ab && aa->layer == ab->layer) {
|
|
int i;
|
|
|
|
res = aa == ab && !aa->is_reversed ? HORIZONTAL_NODE_SPACING : 0;
|
|
for (i = aa->pos_in_layer; i < ab->pos_in_layer; ++i) {
|
|
const RGraphNode *cur = g->layers[aa->layer].nodes[i];
|
|
const RGraphNode *next = g->layers[aa->layer].nodes[i + 1];
|
|
const RANode *anext = get_anode (next);
|
|
const RANode *acur = get_anode (cur);
|
|
int found = false;
|
|
|
|
if (g->dists) {
|
|
d.from = cur;
|
|
d.to = next;
|
|
it = r_list_find (g->dists, &d, (RListComparator)find_dist);
|
|
if (it) {
|
|
struct dist_t *old = (struct dist_t *)r_list_iter_get_data (it);
|
|
res += old->dist;
|
|
found = true;
|
|
}
|
|
}
|
|
|
|
if (!found) {
|
|
int space = HORIZONTAL_NODE_SPACING;
|
|
if (acur->is_reversed && anext->is_reversed) {
|
|
if (!acur->is_reversed)
|
|
res += acur->w / 2;
|
|
else if (!anext->is_reversed)
|
|
res += anext->w / 2;
|
|
res += 1;
|
|
} else {
|
|
res += acur->w / 2 + anext->w / 2 + space;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
/* explictly set the distance between two nodes on the same layer */
|
|
static void set_dist_nodes (const RAGraph *g, int l, int cur, int next) {
|
|
struct dist_t *d, find_el;
|
|
const RGraphNode *vi, *vip;
|
|
const RANode *avi, *avip;
|
|
RListIter *it;
|
|
|
|
if (!g->dists) return;
|
|
vi = g->layers[l].nodes[cur];
|
|
vip = g->layers[l].nodes[next];
|
|
avi = get_anode (vi);
|
|
avip = get_anode (vip);
|
|
|
|
find_el.from = vi;
|
|
find_el.to = vip;
|
|
it = r_list_find (g->dists, &find_el, (RListComparator)find_dist);
|
|
d = it ? (struct dist_t *)r_list_iter_get_data (it) : R_NEW (struct dist_t);
|
|
|
|
d->from = vi;
|
|
d->to = vip;
|
|
d->dist = avip->x - avi->x;
|
|
if (!it)
|
|
r_list_push (g->dists, d);
|
|
}
|
|
|
|
static int is_valid_pos (const RAGraph *g, int l, int pos) {
|
|
return pos >= 0 && pos < g->layers[l].n_nodes;
|
|
}
|
|
|
|
/* computes the set of vertical classes in the graph */
|
|
/* if v is an original node, L(v) = { v }
|
|
* if v is a dummy node, L(v) is the set of all the dummies node that belongs
|
|
* to the same long edge */
|
|
static Sdb *compute_vertical_nodes (const RAGraph *g) {
|
|
Sdb *res = sdb_new0 ();
|
|
int i, j;
|
|
|
|
for (i = 0; i < g->n_layers; ++i) {
|
|
for (j = 0; j < g->layers[i].n_nodes; ++j) {
|
|
RGraphNode *gn = g->layers[i].nodes[j];
|
|
const RList *Ln = hash_get_rlist (res, gn);
|
|
const RANode *an = get_anode (gn);
|
|
|
|
if (!Ln) {
|
|
RList *vert = r_list_new ();
|
|
hash_set (res, gn, vert);
|
|
if (an->is_dummy) {
|
|
RGraphNode *next = gn;
|
|
const RANode *anext = get_anode (next);
|
|
|
|
while (anext->is_dummy) {
|
|
r_list_append (vert, next);
|
|
next = r_graph_nth_neighbour (g->graph, next, 0);
|
|
if (!next) break;
|
|
anext = get_anode (next);
|
|
}
|
|
} else {
|
|
r_list_append (vert, gn);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
/* computes left or right classes, used to place dummies node */
|
|
/* classes respect three properties:
|
|
* - v E C
|
|
* - w E C => L(v) is a subset of C
|
|
* - w E C, the s+(w) exists and is not in any class yet => s+(w) E C */
|
|
static RList **compute_classes (const RAGraph *g, Sdb *v_nodes, int is_left, int *n_classes) {
|
|
int i, j, c;
|
|
RList **res = R_NEWS0 (RList *, g->n_layers);
|
|
RGraphNode *gn;
|
|
const RListIter *it;
|
|
RANode *n;
|
|
|
|
graph_foreach_anode (r_graph_get_nodes (g->graph), it, gn, n) {
|
|
n->klass = -1;
|
|
}
|
|
|
|
for (i = 0; i < g->n_layers; ++i) {
|
|
c = i;
|
|
|
|
for (j = is_left ? 0 : g->layers[i].n_nodes - 1;
|
|
(is_left && j < g->layers[i].n_nodes) || (!is_left && j >= 0);
|
|
j = is_left ? j + 1 : j - 1) {
|
|
const RGraphNode *gj = g->layers[i].nodes[j];
|
|
const RANode *aj = get_anode (gj);
|
|
|
|
if (aj->klass == -1) {
|
|
const RList *laj = hash_get_rlist (v_nodes, gj);
|
|
|
|
if (!res[c])
|
|
res[c] = r_list_new ();
|
|
graph_foreach_anode (laj, it, gn, n) {
|
|
r_list_append (res[c], gn);
|
|
n->klass = c;
|
|
}
|
|
} else {
|
|
c = aj->klass;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (n_classes)
|
|
*n_classes = g->n_layers;
|
|
return res;
|
|
}
|
|
|
|
static int cmp_dist (const size_t a, const size_t b) {
|
|
return (int)a < (int)b;
|
|
}
|
|
|
|
static RGraphNode *get_sibling (const RAGraph *g, const RANode *n, int is_left, int is_adjust_class) {
|
|
RGraphNode *res = NULL;
|
|
int pos = n->pos_in_layer;
|
|
|
|
if ((is_left && is_adjust_class) || (!is_left && !is_adjust_class))
|
|
pos++;
|
|
else pos--;
|
|
|
|
if (is_valid_pos (g, n->layer, pos))
|
|
res = g->layers[n->layer].nodes[pos];
|
|
return res;
|
|
}
|
|
|
|
static int adjust_class_val (const RAGraph *g, const RGraphNode *gn, const RGraphNode *sibl, Sdb *res, int is_left) {
|
|
if (is_left) return hash_get_int (res, sibl) - hash_get_int (res, gn) - dist_nodes (g, gn, sibl);
|
|
return hash_get_int (res, gn) - hash_get_int (res, sibl) - dist_nodes (g, sibl, gn);
|
|
}
|
|
|
|
/* adjusts the position of previously placed left/right classes */
|
|
/* tries to place classes as close as possible */
|
|
static void adjust_class (const RAGraph *g, int is_left, RList **classes, Sdb *res, int c) {
|
|
const RGraphNode *gn;
|
|
const RListIter *it;
|
|
const RANode *an;
|
|
int dist, v, is_first = true;
|
|
|
|
graph_foreach_anode (classes[c], it, gn, an) {
|
|
const RGraphNode *sibling;
|
|
const RANode *sibl_anode;
|
|
|
|
sibling = get_sibling (g, an, is_left, true);
|
|
if (!sibling) continue;
|
|
sibl_anode = get_anode (sibling);
|
|
if (sibl_anode->klass == c) continue;
|
|
v = adjust_class_val (g, gn, sibling, res, is_left);
|
|
dist = is_first ? v : R_MIN (dist, v);
|
|
is_first = false;
|
|
}
|
|
|
|
if (is_first) {
|
|
RList *heap = r_list_new ();
|
|
int len;
|
|
|
|
graph_foreach_anode (classes[c], it, gn, an) {
|
|
const RList *neigh = r_graph_all_neighbours (g->graph, gn);
|
|
const RGraphNode *gk;
|
|
const RListIter *itk;
|
|
const RANode *ak;
|
|
|
|
graph_foreach_anode (neigh, itk, gk, ak) {
|
|
if (ak->klass < c)
|
|
r_list_append (heap, (void *)(size_t)(ak->x - an->x));
|
|
}
|
|
}
|
|
|
|
len = r_list_length (heap);
|
|
if (len == 0) {
|
|
dist = 0;
|
|
} else {
|
|
r_list_sort (heap, (RListComparator)cmp_dist);
|
|
dist = (int)(size_t)r_list_get_n (heap, len / 2);
|
|
}
|
|
|
|
r_list_free (heap);
|
|
}
|
|
|
|
graph_foreach_anode (classes[c], it, gn, an) {
|
|
const int old_val = hash_get_int (res, gn);
|
|
const int new_val = is_left ? old_val + dist : old_val - dist;
|
|
hash_set (res, gn, new_val);
|
|
}
|
|
}
|
|
|
|
static int place_nodes_val (const RAGraph *g, const RGraphNode *gn, const RGraphNode *sibl, Sdb *res, int is_left) {
|
|
if (is_left) return hash_get_int (res, sibl) + dist_nodes (g, sibl, gn);
|
|
return hash_get_int (res, sibl) - dist_nodes (g, gn, sibl);
|
|
}
|
|
|
|
static int place_nodes_sel_p (int newval, int oldval, int is_first, int is_left) {
|
|
if (is_first) return newval;
|
|
if (is_left) return R_MAX (oldval, newval);
|
|
return R_MIN (oldval, newval);
|
|
}
|
|
|
|
/* places left/right the nodes of a class */
|
|
static void place_nodes (const RAGraph *g, const RGraphNode *gn, int is_left, Sdb *v_nodes, RList **classes, Sdb *res, Sdb *placed) {
|
|
const RList *lv = hash_get_rlist (v_nodes, gn);
|
|
int p = 0, v, is_first = true;
|
|
const RGraphNode *gk;
|
|
const RListIter *itk;
|
|
const RANode *ak;
|
|
|
|
graph_foreach_anode (lv, itk, gk, ak) {
|
|
const RGraphNode *sibling;
|
|
const RANode *sibl_anode;
|
|
|
|
sibling = get_sibling (g, ak, is_left, false);
|
|
if (!sibling) continue;
|
|
sibl_anode = get_anode (sibling);
|
|
if (ak->klass == sibl_anode->klass) {
|
|
if (!hash_get (placed, sibling))
|
|
place_nodes (g, sibling, is_left, v_nodes, classes, res, placed);
|
|
|
|
v = place_nodes_val (g, gk, sibling, res, is_left);
|
|
p = place_nodes_sel_p (v, p, is_first, is_left);
|
|
is_first = false;
|
|
}
|
|
}
|
|
|
|
if (is_first)
|
|
p = is_left ? 0 : 50;
|
|
|
|
graph_foreach_anode (lv, itk, gk, ak) {
|
|
hash_set (res, gk, p);
|
|
hash_set (placed, gk, true);
|
|
}
|
|
}
|
|
|
|
/* computes the position to the left/right of all the nodes */
|
|
static Sdb *compute_pos (const RAGraph *g, int is_left, Sdb *v_nodes) {
|
|
Sdb *res, *placed;
|
|
RList **classes;
|
|
int n_classes, i;
|
|
|
|
classes = compute_classes (g, v_nodes, is_left, &n_classes);
|
|
if (!classes) return NULL;
|
|
|
|
res = sdb_new0 ();
|
|
placed = sdb_new0 ();
|
|
for (i = 0; i < n_classes; ++i) {
|
|
const RGraphNode *gn;
|
|
const RListIter *it;
|
|
|
|
r_list_foreach (classes[i], it, gn) {
|
|
if (!hash_get_rnode (placed, gn)) {
|
|
place_nodes (g, gn, is_left, v_nodes, classes, res, placed);
|
|
}
|
|
}
|
|
|
|
adjust_class (g, is_left, classes, res, i);
|
|
}
|
|
|
|
sdb_free (placed);
|
|
for (i = 0; i < n_classes; ++i) {
|
|
if (classes[i])
|
|
r_list_free (classes[i]);
|
|
}
|
|
free (classes);
|
|
return res;
|
|
}
|
|
|
|
/* calculates position of all nodes, but in particular dummies nodes */
|
|
/* computes two different placements (called "left"/"right") and set the final
|
|
* position of each node to the average of the values in the two placements */
|
|
static void place_dummies (const RAGraph *g) {
|
|
const RList *nodes;
|
|
Sdb *xminus, *xplus, *vertical_nodes;
|
|
const RGraphNode *gn;
|
|
const RListIter *it;
|
|
RANode *n;
|
|
|
|
vertical_nodes = compute_vertical_nodes (g);
|
|
if (!vertical_nodes) return;
|
|
xminus = compute_pos (g, true, vertical_nodes);
|
|
if (!xminus) goto xminus_err;
|
|
xplus = compute_pos (g, false, vertical_nodes);
|
|
if (!xplus) goto xplus_err;
|
|
|
|
nodes = r_graph_get_nodes (g->graph);
|
|
graph_foreach_anode (nodes, it, gn, n) {
|
|
n->x = (hash_get_int (xminus, gn) + hash_get_int (xplus, gn)) / 2;
|
|
}
|
|
|
|
sdb_free (xplus);
|
|
xplus_err:
|
|
sdb_free (xminus);
|
|
xminus_err:
|
|
sdb_free (vertical_nodes);
|
|
}
|
|
|
|
static RGraphNode *get_right_dummy (const RAGraph *g, const RGraphNode *n) {
|
|
const RANode *an = get_anode (n);
|
|
int k, layer = an->layer;
|
|
|
|
for (k = an->pos_in_layer + 1; k < g->layers[layer].n_nodes; ++k) {
|
|
RGraphNode *gk = g->layers[layer].nodes[k];
|
|
const RANode *ak = get_anode (gk);
|
|
|
|
if (ak->is_dummy)
|
|
return gk;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void adjust_directions (const RAGraph *g, int i, int from_up, Sdb *D, Sdb *P) {
|
|
const RGraphNode *vm = NULL, *wm = NULL;
|
|
const RANode *vma = NULL, *wma = NULL;
|
|
int j, d = from_up ? 1 : -1;
|
|
|
|
if (i + d < 0 || i + d >= g->n_layers) return;
|
|
|
|
for (j = 0; j < g->layers[i + d].n_nodes; ++j) {
|
|
const RGraphNode *wp, *vp = g->layers[i + d].nodes[j];
|
|
const RANode *wpa, *vpa = get_anode (vp);
|
|
|
|
if (!vpa->is_dummy) continue;
|
|
if (from_up) wp = r_list_get_n (r_graph_innodes (g->graph, vp), 0);
|
|
else wp = r_graph_nth_neighbour (g->graph, vp, 0);
|
|
wpa = get_anode (wp);
|
|
if (!wpa->is_dummy) continue;
|
|
|
|
if (vm) {
|
|
int p = hash_get_int (P, wm);
|
|
int k;
|
|
|
|
for (k = wma->pos_in_layer + 1; k < wpa->pos_in_layer; ++k) {
|
|
const RGraphNode *w = g->layers[wma->layer].nodes[k];
|
|
const RANode *aw = get_anode (w);
|
|
|
|
if (aw->is_dummy)
|
|
p &= hash_get_int (P, w);
|
|
}
|
|
if (p) {
|
|
hash_set (D, vm, from_up);
|
|
for (k = vma->pos_in_layer + 1; k < vpa->pos_in_layer; ++k) {
|
|
const RGraphNode *v = g->layers[vma->layer].nodes[k];
|
|
const RANode *av = get_anode (v);
|
|
|
|
if (av->is_dummy)
|
|
hash_set (D, v, from_up);
|
|
}
|
|
}
|
|
}
|
|
|
|
vm = vp;
|
|
wm = wp;
|
|
vma = get_anode (vm);
|
|
wma = get_anode (wm);
|
|
}
|
|
}
|
|
|
|
/* find a placement for a single node */
|
|
static void place_single (const RAGraph *g, int l, const RGraphNode *bm,
|
|
const RGraphNode *bp, int from_up, int va) {
|
|
const RGraphNode *gk, *v = g->layers[l].nodes[va];
|
|
const RANode *ak;
|
|
RANode *av = get_anode (v);
|
|
const RList *neigh;
|
|
const RListIter *itk;
|
|
int len;
|
|
|
|
if (from_up)
|
|
neigh = r_graph_innodes (g->graph, v);
|
|
else
|
|
neigh = r_graph_get_neighbours (g->graph, v);
|
|
|
|
len = r_list_length (neigh);
|
|
if (len == 0)
|
|
return;
|
|
|
|
int sum_x = 0;
|
|
graph_foreach_anode (neigh, itk, gk, ak) {
|
|
if (ak->is_reversed) {
|
|
len--;
|
|
continue;
|
|
}
|
|
|
|
sum_x += ak->x;
|
|
}
|
|
|
|
if (len == 0)
|
|
return;
|
|
|
|
av->x = sum_x / len;
|
|
if (bm) {
|
|
const RANode *bma = get_anode (bm);
|
|
av->x = R_MAX (av->x, bma->x + dist_nodes (g, bm, v));
|
|
}
|
|
if (bp) {
|
|
const RANode *bpa = get_anode (bp);
|
|
av->x = R_MIN (av->x, bpa->x - dist_nodes (g, v, bp));
|
|
}
|
|
}
|
|
|
|
static int RM_listcmp (const struct len_pos_t *a, const struct len_pos_t *b) {
|
|
return a->pos < b->pos;
|
|
}
|
|
|
|
static int RP_listcmp (const struct len_pos_t *a, const struct len_pos_t *b) {
|
|
return a->pos >= b->pos;
|
|
}
|
|
|
|
static void collect_changes (const RAGraph *g, int l, const RGraphNode *b, int from_up, int s, int e, RList *list, int is_left) {
|
|
const RGraphNode *vt = g->layers[l].nodes[e - 1];
|
|
const RGraphNode *vtp = g->layers[l].nodes[s];
|
|
RListComparator lcmp;
|
|
struct len_pos_t *cx;
|
|
int i;
|
|
|
|
lcmp = is_left ? (RListComparator)RM_listcmp : (RListComparator)RP_listcmp;
|
|
|
|
for (i = is_left ? s : e - 1;
|
|
(is_left && i < e) || (!is_left && i >= s);
|
|
i = is_left ? i + 1 : i - 1) {
|
|
const RGraphNode *v, *vi = g->layers[l].nodes[i];
|
|
const RANode *av, *avi = get_anode (vi);
|
|
const RList *neigh;
|
|
const RListIter *it;
|
|
int c = 0;
|
|
|
|
if (!avi) continue;
|
|
neigh = from_up
|
|
? r_graph_innodes (g->graph, vi)
|
|
: r_graph_get_neighbours (g->graph, vi);
|
|
|
|
graph_foreach_anode (neigh, it, v, av) {
|
|
if ((is_left && av->x >= avi->x) || (!is_left && av->x <= avi->x)) {
|
|
c++;
|
|
} else {
|
|
cx = R_NEW (struct len_pos_t);
|
|
|
|
c--;
|
|
cx->len = 2;
|
|
cx->pos = av->x;
|
|
if (is_left)
|
|
cx->pos += dist_nodes (g, vi, vt);
|
|
else
|
|
cx->pos -= dist_nodes (g, vtp, vi);
|
|
r_list_add_sorted (list, cx, lcmp);
|
|
}
|
|
}
|
|
|
|
cx = R_NEW0 (struct len_pos_t);
|
|
cx->len = c;
|
|
cx->pos = avi->x;
|
|
if (is_left) cx->pos += dist_nodes (g, vi, vt);
|
|
else cx->pos -= dist_nodes (g, vtp, vi);
|
|
r_list_add_sorted (list, cx, lcmp);
|
|
}
|
|
|
|
if (b) {
|
|
const RANode *ab = get_anode (b);
|
|
|
|
cx = R_NEW (struct len_pos_t);
|
|
cx->len = is_left ? INT_MAX : INT_MIN;
|
|
cx->pos = ab->x;
|
|
if (is_left) cx->pos += dist_nodes (g, b, vt);
|
|
else cx->pos -= dist_nodes (g, vtp, b);
|
|
r_list_add_sorted (list, cx, lcmp);
|
|
}
|
|
}
|
|
|
|
static void combine_sequences (const RAGraph *g, int l, const RGraphNode *bm, const RGraphNode *bp, int from_up, int a, int r) {
|
|
RList *Rm = r_list_new (), *Rp = r_list_new ();
|
|
const RGraphNode *vt, *vtp;
|
|
RANode *at, *atp;
|
|
int rm, rp, t, m, i;
|
|
|
|
t = (a + r) / 2;
|
|
vt = g->layers[l].nodes[t - 1];
|
|
vtp = g->layers[l].nodes[t];
|
|
at = get_anode (vt);
|
|
atp = get_anode (vtp);
|
|
|
|
collect_changes (g, l, bm, from_up, a, t, Rm, true);
|
|
collect_changes (g, l, bp, from_up, t, r, Rp, false);
|
|
rm = rp = 0;
|
|
|
|
m = dist_nodes (g, vt, vtp);
|
|
while (atp->x - at->x < m) {
|
|
if (atp->x == at->x) {
|
|
int step = m / 2;
|
|
at->x -= step;
|
|
atp->x += m - step;
|
|
} else {
|
|
if (rm < rp) {
|
|
if (r_list_empty (Rm)) {
|
|
at->x = atp->x - m;
|
|
} else {
|
|
struct len_pos_t *cx = (struct len_pos_t *)r_list_pop (Rm);
|
|
rm = rm + cx->len;
|
|
at->x = R_MAX (cx->pos, atp->x - m);
|
|
free (cx);
|
|
}
|
|
} else {
|
|
if (r_list_empty (Rp)) {
|
|
atp->x = at->x + m;
|
|
} else {
|
|
struct len_pos_t *cx = (struct len_pos_t *)r_list_pop (Rp);
|
|
rp = rp + cx->len;
|
|
atp->x = R_MIN (cx->pos, at->x + m);
|
|
free (cx);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
r_list_free (Rm);
|
|
r_list_free (Rp);
|
|
|
|
for (i = t - 2; i >= a; --i) {
|
|
const RGraphNode *gv = g->layers[l].nodes[i];
|
|
RANode *av = get_anode (gv);
|
|
av->x = R_MIN (av->x, at->x - dist_nodes (g, gv, vt));
|
|
}
|
|
|
|
for (i = t + 1; i < r; ++i) {
|
|
const RGraphNode *gv = g->layers[l].nodes[i];
|
|
RANode *av = get_anode (gv);
|
|
av->x = R_MAX (av->x, atp->x + dist_nodes (g, vtp, gv));
|
|
}
|
|
}
|
|
|
|
/* places a sequence of consecutive original nodes */
|
|
/* it tries to minimize the distance between each node in the sequence and its
|
|
* neighbours in the "previous" layer. Those neighbours are considered as
|
|
* "fixed". The previous layer depends on the direction used during the layers
|
|
* traversal */
|
|
static void place_sequence (const RAGraph *g, int l, const RGraphNode *bm, const RGraphNode *bp, int from_up, int va, int vr) {
|
|
int vt;
|
|
|
|
if (vr == va + 1) {
|
|
place_single (g, l, bm, bp, from_up, va);
|
|
} else if (vr > va + 1) {
|
|
vt = (vr + va) / 2;
|
|
place_sequence (g, l, bm, bp, from_up, va, vt);
|
|
place_sequence (g, l, bm, bp, from_up, vt, vr);
|
|
combine_sequences (g, l, bm, bp, from_up, va, vr);
|
|
}
|
|
}
|
|
|
|
/* finds the placements of nodes while traversing the graph in the given
|
|
* direction */
|
|
/* places all the sequences of consecutive original nodes in each layer. */
|
|
static void original_traverse_l (const RAGraph *g, Sdb *D, Sdb *P, int from_up) {
|
|
int i, k, va, vr;
|
|
|
|
for (i = from_up ? 0 : g->n_layers - 1;
|
|
(from_up && i < g->n_layers) || (!from_up && i >= 0);
|
|
i = from_up ? i + 1 : i - 1) {
|
|
int j;
|
|
const RGraphNode *bm = NULL;
|
|
const RANode *bma = NULL;
|
|
|
|
j = 0;
|
|
while (j < g->layers[i].n_nodes && !bm) {
|
|
const RGraphNode *gn = g->layers[i].nodes[j];
|
|
const RANode *an = get_anode (gn);
|
|
|
|
if (an->is_dummy) {
|
|
va = 0;
|
|
vr = j;
|
|
bm = gn;
|
|
bma = an;
|
|
}
|
|
j++;
|
|
}
|
|
|
|
if (!bm) {
|
|
va = 0;
|
|
vr = g->layers[i].n_nodes;
|
|
}
|
|
|
|
place_sequence (g, i, NULL, bm, from_up, va, vr);
|
|
for (k = va; k < vr - 1; ++k)
|
|
set_dist_nodes (g, i, k, k + 1);
|
|
|
|
if (is_valid_pos (g, i, vr - 1) && bm)
|
|
set_dist_nodes (g, i, vr - 1, bma->pos_in_layer);
|
|
|
|
while (bm) {
|
|
const RGraphNode *bp = get_right_dummy (g, bm);
|
|
const RANode *bpa = NULL;
|
|
bma = get_anode (bm);
|
|
|
|
if (!bp) {
|
|
va = bma->pos_in_layer + 1;
|
|
vr = g->layers[bma->layer].n_nodes;
|
|
place_sequence (g, i, bm, NULL, from_up, va, vr);
|
|
for (k = va; k < vr - 1; ++k)
|
|
set_dist_nodes (g, i, k, k + 1);
|
|
|
|
if (is_valid_pos (g, i, va))
|
|
set_dist_nodes (g, i, bma->pos_in_layer, va);
|
|
} else if (hash_get_int (D, bm) == from_up) {
|
|
bpa = get_anode (bp);
|
|
va = bma->pos_in_layer + 1;
|
|
vr = bpa->pos_in_layer;
|
|
place_sequence (g, i, bm, bp, from_up, va, vr);
|
|
hash_set (P, bm, true);
|
|
}
|
|
|
|
bm = bp;
|
|
}
|
|
|
|
adjust_directions (g, i, from_up, D, P);
|
|
}
|
|
}
|
|
|
|
/* computes a final position of original nodes, considering dummies nodes as
|
|
* fixed */
|
|
/* set the node placements traversing the graph downward and then upward */
|
|
static void place_original (RAGraph *g) {
|
|
const RList *nodes = r_graph_get_nodes (g->graph);
|
|
Sdb *D, *P;
|
|
const RGraphNode *gn;
|
|
const RListIter *itn;
|
|
const RANode *an;
|
|
|
|
D = sdb_new0 ();
|
|
P = sdb_new0 ();
|
|
g->dists = r_list_new ();
|
|
g->dists->free = (RListFree)free;
|
|
|
|
graph_foreach_anode (nodes, itn, gn, an) {
|
|
if (!an->is_dummy) continue;
|
|
const RGraphNode *right_v = get_right_dummy (g, gn);
|
|
if (right_v) {
|
|
const RANode *right = get_anode (right_v);
|
|
|
|
hash_set (D, gn, 0);
|
|
int dt_eq = right->x - an->x == dist_nodes (g, gn, right_v);
|
|
hash_set (P, gn, dt_eq);
|
|
}
|
|
}
|
|
|
|
original_traverse_l (g, D, P, true);
|
|
original_traverse_l (g, D, P, false);
|
|
|
|
r_list_free (g->dists);
|
|
g->dists = NULL;
|
|
sdb_free (P);
|
|
sdb_free (D);
|
|
}
|
|
|
|
static void restore_original_edges (const RAGraph *g) {
|
|
const RListIter *it;
|
|
const RGraphEdge *e;
|
|
|
|
r_list_foreach (g->long_edges, it, e) {
|
|
RANode *from, *to;
|
|
from = e->from ? get_anode (e->from) : NULL;
|
|
to = e->to ? get_anode (e->to) : NULL;
|
|
r_agraph_add_edge_at (g, from, to, e->nth);
|
|
}
|
|
|
|
r_list_foreach (g->back_edges, it, e) {
|
|
RANode *from, *to;
|
|
from = e->from ? get_anode (e->from) : NULL;
|
|
to = e->to ? get_anode (e->to) : NULL;
|
|
r_agraph_del_edge (g, to, from);
|
|
r_agraph_add_edge_at (g, from, to, e->nth);
|
|
}
|
|
}
|
|
|
|
static void create_edge_from_dummies (const RAGraph *g, RANode *an, RList *toremove) {
|
|
RGraphNode *n = an->gnode;
|
|
RGraphNode *from = r_list_get_n (r_graph_innodes (g->graph, n), 0);
|
|
RANode *a_from = get_anode (from);
|
|
RListIter *(*add_to_list)(RList *, void *) = NULL;
|
|
AEdge *e = R_NEW0 (AEdge);
|
|
|
|
e->x = r_list_new ();
|
|
e->y = r_list_new ();
|
|
e->is_reversed = an->is_reversed;
|
|
if (e->is_reversed) {
|
|
e->to = a_from;
|
|
add_to_list = r_list_prepend;
|
|
add_to_list (e->x, (void *)(size_t)an->x);
|
|
add_to_list (e->y, (void *)(size_t)a_from->y);
|
|
} else {
|
|
e->from = a_from;
|
|
add_to_list = r_list_append;
|
|
}
|
|
|
|
while (an->is_dummy) {
|
|
add_to_list (toremove, n);
|
|
|
|
add_to_list (e->x, (void *)(size_t)an->x);
|
|
add_to_list (e->y, (void *)(size_t)an->y);
|
|
|
|
add_to_list (e->x, (void *)(size_t)an->x);
|
|
add_to_list (e->y, (void *)(size_t)
|
|
(an->y + g->layers[an->layer].height));
|
|
|
|
n = r_graph_nth_neighbour (g->graph, n, 0);
|
|
an = get_anode (n);
|
|
}
|
|
|
|
if (e->is_reversed) {
|
|
e->from = an;
|
|
} else {
|
|
e->to = an;
|
|
}
|
|
r_list_append (g->edges, e);
|
|
}
|
|
|
|
static void analyze_back_edges (const RAGraph *g, RANode *an) {
|
|
const RList *neigh = r_graph_get_neighbours (g->graph, an->gnode);
|
|
RListIter *itk;
|
|
RGraphNode *gk;
|
|
RANode *ak;
|
|
int j = 0, i = -1;
|
|
|
|
/* traverse all neighbours and analyze only the ones that create back
|
|
* edges. */
|
|
graph_foreach_anode (neigh, itk, gk, ak) {
|
|
RGraphNode *fn, *ln;
|
|
RANode *first, *last;
|
|
const RList *tp;
|
|
AEdge *e;
|
|
|
|
i++;
|
|
if (ak->layer > an->layer) continue;
|
|
e = R_NEW0 (AEdge);
|
|
e->is_reversed = true;
|
|
e->from = an;
|
|
e->to = ak;
|
|
e->x = r_list_new ();
|
|
e->y = r_list_new ();
|
|
|
|
tp = r_graph_get_neighbours (g->graph, ak->gnode);
|
|
if (r_list_length (tp) > 0) {
|
|
fn = r_list_get_bottom (tp);
|
|
ln = r_list_get_top (tp);
|
|
first = get_anode (fn);
|
|
last = get_anode (ln);
|
|
|
|
if (first == an) {
|
|
r_list_append (e->x,
|
|
(void *)(size_t)(an->x - 2 - j));
|
|
r_list_append (e->y, (void *)(size_t)ak->y);
|
|
} else {
|
|
r_list_append (e->x,
|
|
(void *)(size_t)(last->x + last->w + 2 + j));
|
|
r_list_append (e->y, (void *)(size_t)ak->y);
|
|
}
|
|
}
|
|
r_list_append (g->edges, e);
|
|
j++;
|
|
}
|
|
}
|
|
|
|
static void remove_dummy_nodes (const RAGraph *g) {
|
|
RGraphNode *gn;
|
|
const RListIter *it;
|
|
RList *toremove = r_list_new ();
|
|
int i, j;
|
|
|
|
/* traverse all dummy nodes to keep track
|
|
* of the path long edges should go by. */
|
|
for (i = 0; i < g->n_layers; ++i) {
|
|
for (j = 0; j < g->layers[i].n_nodes; ++j) {
|
|
RGraphNode *n = g->layers[i].nodes[j];
|
|
RANode *an = get_anode (n);
|
|
if (an->is_dummy && !r_list_contains (toremove, n)) {
|
|
create_edge_from_dummies (g, an, toremove);
|
|
} else if (!an->is_dummy) {
|
|
analyze_back_edges (g, an);
|
|
}
|
|
}
|
|
}
|
|
|
|
r_list_foreach (toremove, it, gn) {
|
|
r_graph_del_node (g->graph, gn);
|
|
}
|
|
|
|
r_list_free (toremove);
|
|
}
|
|
|
|
/* 1) trasform the graph into a DAG
|
|
* 2) partition the nodes in layers
|
|
* 3) split long edges that traverse multiple layers
|
|
* 4) reorder nodes in each layer to reduce the number of edge crossing
|
|
* 5) assign x and y coordinates to each node
|
|
* 6) restore the original graph, with long edges and cycles */
|
|
static void set_layout(RAGraph *g) {
|
|
int i, j, k;
|
|
|
|
if (g->edges) r_list_free (g->edges);
|
|
g->edges = r_list_new ();
|
|
|
|
remove_cycles (g);
|
|
assign_layers (g);
|
|
create_dummy_nodes (g);
|
|
create_layers (g);
|
|
minimize_crossings (g);
|
|
|
|
/* identify row height */
|
|
for (i = 0; i < g->n_layers; i++) {
|
|
int rh = 0;
|
|
for (j = 0; j < g->layers[i].n_nodes; ++j) {
|
|
const RANode *n = get_anode (g->layers[i].nodes[j]);
|
|
if (n->h > rh)
|
|
rh = n->h;
|
|
}
|
|
g->layers[i].height = rh;
|
|
}
|
|
|
|
/* x-coordinate assignment: algorithm based on:
|
|
* A Fast Layout Algorithm for k-Level Graphs
|
|
* by C. Buchheim, M. Junger, S. Leipert */
|
|
place_dummies (g);
|
|
place_original (g);
|
|
|
|
/* vertical align */
|
|
for (i = 0; i < g->n_layers; ++i) {
|
|
for (j = 0; j < g->layers[i].n_nodes; ++j) {
|
|
RANode *n = get_anode (g->layers[i].nodes[j]);
|
|
n->y = 1;
|
|
for (k = 0; k < n->layer; ++k) {
|
|
n->y += g->layers[k].height + VERTICAL_NODE_SPACING;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* finalize x coordinate */
|
|
for (i = 0; i < g->n_layers; ++i) {
|
|
for (j = 0; j < g->layers[i].n_nodes; ++j) {
|
|
RANode *n = get_anode (g->layers[i].nodes[j]);
|
|
n->x -= n->w / 2;
|
|
}
|
|
}
|
|
|
|
restore_original_edges (g);
|
|
remove_dummy_nodes (g);
|
|
|
|
/* free all temporary structures used during layout */
|
|
for (i = 0; i < g->n_layers; ++i)
|
|
free (g->layers[i].nodes);
|
|
free (g->layers);
|
|
r_list_free (g->long_edges);
|
|
r_list_free (g->back_edges);
|
|
}
|
|
|
|
static char *get_body (RCore *core, ut64 addr, int size, bool with_offset,
|
|
bool is_esil, bool disasm_insts) {
|
|
char *body;
|
|
int o_fcnlines = r_config_get_i (core->config, "asm.fcnlines");
|
|
int o_lines = r_config_get_i (core->config, "asm.lines");
|
|
int o_bytes = r_config_get_i (core->config, "asm.bytes");
|
|
int o_cmtcol = r_config_get_i (core->config, "asm.cmtcol");
|
|
int o_marks = r_config_get_i (core->config, "asm.marks");
|
|
int o_offset = r_config_get_i (core->config, "asm.offset");
|
|
int o_esil = r_config_get_i (core->config, "asm.esil");
|
|
int o_cursor = core->print->cur_enabled;
|
|
const char *cmd = disasm_insts ? "pd" : "pD";
|
|
// configure options
|
|
r_config_set_i (core->config, "asm.fcnlines", false);
|
|
r_config_set_i (core->config, "asm.lines", false);
|
|
r_config_set_i (core->config, "asm.cmtcol", 0);
|
|
r_config_set_i (core->config, "asm.marks", false);
|
|
r_config_set_i (core->config, "asm.esil", is_esil);
|
|
core->print->cur_enabled = false;
|
|
|
|
if (with_offset) {
|
|
r_config_set_i (core->config, "asm.offset", true);
|
|
r_config_set_i (core->config, "asm.bytes", true);
|
|
} else {
|
|
r_config_set_i (core->config, "asm.bytes", false);
|
|
r_config_set_i (core->config, "asm.offset", false);
|
|
}
|
|
body = r_core_cmd_strf (core,
|
|
"%s %d @ 0x%08"PFMT64x, cmd, size, addr);
|
|
// restore original options
|
|
core->print->cur_enabled = o_cursor;
|
|
r_config_set_i (core->config, "asm.esil", o_esil);
|
|
r_config_set_i (core->config, "asm.fcnlines", o_fcnlines);
|
|
r_config_set_i (core->config, "asm.lines", o_lines);
|
|
r_config_set_i (core->config, "asm.bytes", o_bytes);
|
|
r_config_set_i (core->config, "asm.cmtcol", o_cmtcol);
|
|
r_config_set_i (core->config, "asm.marks", o_marks);
|
|
r_config_set_i (core->config, "asm.offset", o_offset);
|
|
return body;
|
|
}
|
|
|
|
static void get_bbupdate(RAGraph *g, RCore *core, RAnalFunction *fcn) {
|
|
RAnalBlock *bb;
|
|
RListIter *iter;
|
|
core->keep_asmqjmps = false;
|
|
r_list_foreach (fcn->bbs, iter, bb) {
|
|
RANode *node;
|
|
char *title, *body;
|
|
|
|
if (bb->addr == UT64_MAX)
|
|
continue;
|
|
|
|
body = get_body (core, bb->addr, bb->size, is_offset (g),
|
|
is_esil (g), false);
|
|
title = get_title (bb->addr);
|
|
node = r_agraph_get_node (g, title);
|
|
if (node) {
|
|
free (node->body);
|
|
node->body = body;
|
|
} else {
|
|
free (body);
|
|
}
|
|
free (title);
|
|
core->keep_asmqjmps = true;
|
|
}
|
|
}
|
|
|
|
/* build the RGraph inside the RAGraph g, starting from the Basic Blocks */
|
|
static int get_bbnodes(RAGraph *g, RCore *core, RAnalFunction *fcn) {
|
|
RAnalBlock *bb;
|
|
RListIter *iter;
|
|
|
|
core->keep_asmqjmps = false;
|
|
r_list_foreach (fcn->bbs, iter, bb) {
|
|
RANode *node;
|
|
char *title, *body;
|
|
|
|
if (bb->addr == UT64_MAX) continue;
|
|
|
|
body = get_body (core, bb->addr, bb->size, is_offset (g),
|
|
is_esil (g), false);
|
|
title = get_title (bb->addr);
|
|
|
|
node = r_agraph_add_node (g, title, body);
|
|
free (body);
|
|
free (title);
|
|
if (!node) return false;
|
|
core->keep_asmqjmps = true;
|
|
}
|
|
|
|
r_list_foreach (fcn->bbs, iter, bb) {
|
|
RANode *u, *v;
|
|
char *title;
|
|
|
|
if (bb->addr == UT64_MAX)
|
|
continue;
|
|
|
|
title = get_title (bb->addr);
|
|
u = r_agraph_get_node (g, title);
|
|
free (title);
|
|
if (bb->jump != UT64_MAX) {
|
|
title = get_title (bb->jump);
|
|
v = r_agraph_get_node (g, title);
|
|
free (title);
|
|
r_agraph_add_edge (g, u, v);
|
|
}
|
|
if (bb->fail != UT64_MAX) {
|
|
title = get_title (bb->fail);
|
|
v = r_agraph_get_node (g, title);
|
|
free (title);
|
|
r_agraph_add_edge (g, u, v);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/* build the RGraph inside the RAGraph g, starting from the Call Graph
|
|
* information */
|
|
static int get_cgnodes(RAGraph *g, RCore *core, RAnalFunction *fcn) {
|
|
#if FCN_OLD
|
|
RAnalFunction *f = r_anal_get_fcn_in (core->anal, core->offset, 0);
|
|
RANode *node, *fcn_anode;
|
|
RListIter *iter;
|
|
RAnalRef *ref;
|
|
char *title, *body;
|
|
|
|
if (!f) return false;
|
|
|
|
r_core_seek (core, f->addr, 1);
|
|
|
|
title = get_title (fcn->addr);
|
|
fcn_anode = r_agraph_add_node (g, title, "");
|
|
|
|
free (title);
|
|
if (!fcn_anode) {
|
|
return false;
|
|
}
|
|
|
|
fcn_anode->x = 10;
|
|
fcn_anode->y = 3;
|
|
|
|
r_list_foreach (fcn->refs, iter, ref) {
|
|
/* XXX: something is broken, why there are duplicated
|
|
* nodes here?! goto check fcn->refs!! */
|
|
/* avoid dups wtf */
|
|
title = get_title (ref->addr);
|
|
if (r_agraph_get_node (g, title) != NULL)
|
|
continue;
|
|
free (title);
|
|
|
|
body = get_body (core, ref->addr, 4, is_offset (g),
|
|
is_esil (g), true);
|
|
body = r_str_concat (body, "...\n");
|
|
title = get_title (ref->addr);
|
|
|
|
node = r_agraph_add_node (g, title, body);
|
|
if (!node) return false;
|
|
|
|
free (title);
|
|
free (body);
|
|
|
|
node->x = 10;
|
|
node->y = 10;
|
|
|
|
r_agraph_add_edge (g, fcn_anode, node);
|
|
}
|
|
#else
|
|
eprintf ("Must be sdbized\n");
|
|
#endif
|
|
|
|
return true;
|
|
}
|
|
|
|
static int reload_nodes(RAGraph *g, RCore *core, RAnalFunction *fcn) {
|
|
int is_c = g->is_callgraph;
|
|
return is_c ? get_cgnodes (g, core, fcn) : get_bbnodes (g, core, fcn);
|
|
}
|
|
|
|
static void update_seek(RConsCanvas *can, RANode *n, int force) {
|
|
int x, y, w, h;
|
|
int doscroll = false;
|
|
|
|
if (!n) return;
|
|
|
|
x = n->x + can->sx;
|
|
y = n->y + can->sy;
|
|
w = can->w;
|
|
h = can->h;
|
|
|
|
doscroll = force || y < 0 || y + 5 > h || x + 5 > w || x + n->w + 5 < 0;
|
|
|
|
if (doscroll) {
|
|
can->sx = -n->x - n->w / 2 + w / 2;
|
|
can->sy = -n->y - n->h / 8 + h / 4;
|
|
}
|
|
}
|
|
|
|
static int is_near (const RANode *n, int x, int y, int is_next) {
|
|
if (is_next)
|
|
return (n->y == y && n->x > x) || n->y > y;
|
|
else
|
|
return (n->y == y && n->x < x) || n->y < y;
|
|
}
|
|
|
|
static const RGraphNode *find_near_of (const RAGraph *g, const RGraphNode *cur, int is_next) {
|
|
/* XXX: it's slow */
|
|
const RList *nodes = r_graph_get_nodes (g->graph);
|
|
const RListIter *it;
|
|
const RGraphNode *gn, *resgn = NULL;
|
|
const RANode *n, *acur = cur ? get_anode (cur) : NULL;
|
|
int default_v = is_next ? INT_MIN : INT_MAX;
|
|
int start_y = acur ? acur->y : default_v;
|
|
int start_x = acur ? acur->x : default_v;
|
|
|
|
graph_foreach_anode (nodes, it, gn, n) {
|
|
if (is_near (n, start_x, start_y, is_next)) {
|
|
const RANode *resn;
|
|
|
|
if (!resgn) {
|
|
resgn = gn;
|
|
continue;
|
|
}
|
|
|
|
resn = get_anode (resgn);
|
|
if ((is_next && resn->y > n->y) || (!is_next && resn->y < n->y))
|
|
resgn = gn;
|
|
else if ((is_next && resn->y == n->y && resn->x > n->x) ||
|
|
(!is_next && resn->y == n->y && resn->x < n->x))
|
|
resgn = gn;
|
|
}
|
|
}
|
|
|
|
if (!resgn && cur)
|
|
resgn = find_near_of (g, NULL, is_next);
|
|
|
|
return resgn;
|
|
}
|
|
|
|
static void update_graph_sizes (RAGraph *g) {
|
|
RListIter *it;
|
|
RGraphNode *gk;
|
|
RANode *ak, *min_gn, *max_gn;
|
|
int max_x, max_y;
|
|
int delta_x, delta_y;
|
|
AEdge *e;
|
|
|
|
g->x = g->y = INT_MAX;
|
|
max_x = max_y = INT_MIN;
|
|
min_gn = max_gn = NULL;
|
|
|
|
graph_foreach_anode (r_graph_get_nodes (g->graph), it, gk, ak) {
|
|
if (ak->x < g->x) g->x = ak->x;
|
|
if (ak->y < g->y) {
|
|
g->y = ak->y;
|
|
min_gn = ak;
|
|
}
|
|
if (ak->x + ak->w > max_x) max_x = ak->x + ak->w;
|
|
if (ak->y + ak->h > max_y) {
|
|
max_y = ak->y + ak->h;
|
|
max_gn = ak;
|
|
}
|
|
}
|
|
|
|
/* while calculating the graph size, take into account long edges */
|
|
r_list_foreach (g->edges, it, e) {
|
|
RListIter *kt;
|
|
void *vv;
|
|
int v;
|
|
|
|
r_list_foreach (e->x, kt, vv) {
|
|
v = (int)(size_t)vv;
|
|
if (v < g->x) g->x = v;
|
|
if (v + 1 > max_x) max_x = v + 1;
|
|
}
|
|
r_list_foreach (e->y, kt, vv) {
|
|
v = (int)(size_t)vv;
|
|
if (v < g->y) g->y = v;
|
|
if (v + 1 > max_y) max_y = v + 1;
|
|
}
|
|
}
|
|
|
|
if (min_gn) {
|
|
const RList *neigh = r_graph_innodes (g->graph, min_gn->gnode);
|
|
if (r_list_length (neigh) > 0) {
|
|
g->y--;
|
|
max_y++;
|
|
}
|
|
if (max_gn) {
|
|
const RList *neigh = r_graph_get_neighbours (g->graph, min_gn->gnode);
|
|
if (r_list_length (neigh) > 0)
|
|
max_y++;
|
|
}
|
|
}
|
|
|
|
if (g->x != INT_MAX && g->y != INT_MAX) {
|
|
g->w = max_x - g->x;
|
|
if (g->title) {
|
|
size_t len = strlen (g->title);
|
|
if (len > INT_MAX) g->w = INT_MAX;
|
|
if ((int)len > g->w) g->w = len;
|
|
}
|
|
g->h = max_y - g->y;
|
|
} else {
|
|
g->x = g->y = 0;
|
|
g->w = g->h = 0;
|
|
}
|
|
|
|
sdb_num_set (g->db, "agraph.w", g->w, 0);
|
|
sdb_num_set (g->db, "agraph.h", g->h, 0);
|
|
/* delta_x, delta_y are needed to make every other x,y coordinates
|
|
* unsigned, so that we can use sdb_num_ API */
|
|
delta_x = g->x < 0 ? -g->x : 0;
|
|
delta_y = g->y < 0 ? -g->y : 0;
|
|
sdb_num_set (g->db, "agraph.delta_x", delta_x, 0);
|
|
sdb_num_set (g->db, "agraph.delta_y", delta_y, 0);
|
|
}
|
|
|
|
R_API void r_agraph_set_curnode(RAGraph *g, RANode *a) {
|
|
if (!a) return;
|
|
g->curnode = a->gnode;
|
|
if (a->title) {
|
|
sdb_set (g->db, "agraph.curnode", a->title, 0);
|
|
if (g->on_curnode_change) {
|
|
g->on_curnode_change (a, g->on_curnode_change_data);
|
|
}
|
|
}
|
|
}
|
|
|
|
static ut64 rebase (RAGraph *g, int v) {
|
|
return g->x < 0 ? -g->x + v : v;
|
|
}
|
|
|
|
static void agraph_set_layout(RAGraph *g, bool is_interactive) {
|
|
RListIter *it;
|
|
RGraphNode *n;
|
|
RANode *a;
|
|
|
|
set_layout (g);
|
|
|
|
update_graph_sizes (g);
|
|
graph_foreach_anode (r_graph_get_nodes (g->graph), it, n, a) {
|
|
const char *k;
|
|
k = sdb_fmt (1, "agraph.nodes.%s.x", a->title);
|
|
sdb_num_set (g->db, k, rebase (g, a->x), 0);
|
|
k = sdb_fmt (1, "agraph.nodes.%s.y", a->title);
|
|
sdb_num_set (g->db, k, rebase (g, a->y), 0);
|
|
k = sdb_fmt (1, "agraph.nodes.%s.w", a->title);
|
|
sdb_num_set (g->db, k, a->w, 0);
|
|
k = sdb_fmt (1, "agraph.nodes.%s.h", a->title);
|
|
sdb_num_set (g->db, k, a->h, 0);
|
|
}
|
|
}
|
|
|
|
/* set the willing to center the screen on a particular node */
|
|
static void agraph_update_seek(RAGraph *g, RANode *n, int force) {
|
|
g->update_seek_on = n;
|
|
g->force_update_seek = force;
|
|
}
|
|
|
|
static void agraph_print_node(const RAGraph *g, RANode *n) {
|
|
const int cur = g->curnode && get_anode (g->curnode) == n;
|
|
|
|
if (is_mini (g)) {
|
|
small_RANode_print (g, n, cur);
|
|
} else {
|
|
normal_RANode_print (g, n, cur);
|
|
}
|
|
}
|
|
|
|
static void agraph_print_nodes(const RAGraph *g) {
|
|
const RList *nodes = r_graph_get_nodes (g->graph);
|
|
RGraphNode *gn;
|
|
RListIter *it;
|
|
RANode *n;
|
|
|
|
graph_foreach_anode (nodes, it, gn, n) {
|
|
if (gn != g->curnode) {
|
|
agraph_print_node (g, n);
|
|
}
|
|
}
|
|
|
|
/* draw current node now to make it appear on top */
|
|
if (g->curnode) agraph_print_node (g, get_anode (g->curnode));
|
|
}
|
|
|
|
static int find_ascii_edge (const AEdge *a, const AEdge *b) {
|
|
return a->from == b->from && a->to == b->to ? 0 : 1;
|
|
}
|
|
|
|
/* print an edge between two nodes.
|
|
* nth: specifies if the edge is the true(1)/false(2) branch or if it's the
|
|
* only edge for that node(0), so that a different style will be applied
|
|
* to the drawn line */
|
|
static void agraph_print_edge(const RAGraph *g, RANode *a, RANode *b, int nth) {
|
|
int x, y, x2, y2;
|
|
int xinc;
|
|
RListIter *it;
|
|
AEdge e, *edg = NULL;
|
|
int is_first = true;
|
|
RCanvasLineStyle style;
|
|
|
|
xinc = 4 + 2 * (nth + 1);
|
|
x = a->x + xinc;
|
|
y = a->y + a->h;
|
|
if (nth > 1) nth = 1;
|
|
|
|
switch (nth) {
|
|
case 0: style.color = LINE_TRUE; break;
|
|
case 1: style.color = LINE_FALSE; break;
|
|
case -1: style.color = LINE_UNCJMP; break;
|
|
default: style.color = LINE_NONE; break;
|
|
}
|
|
style.symbol = style.color;
|
|
|
|
e.from = a;
|
|
e.to = b;
|
|
it = r_list_find (g->edges, &e, (RListComparator)find_ascii_edge);
|
|
if (it) {
|
|
int i, len;
|
|
|
|
edg = r_list_iter_get_data (it);
|
|
len = r_list_length (edg->x);
|
|
|
|
for (i = 0; i < len; ++i) {
|
|
x2 = (int)(size_t)r_list_get_n (edg->x, i);
|
|
y2 = (int)(size_t)r_list_get_n (edg->y, i) - 1;
|
|
|
|
if (is_first && nth == 0 && x2 > x) {
|
|
xinc += 4;
|
|
x += 4;
|
|
}
|
|
r_cons_canvas_line (g->can, x, y, x2, y2, &style);
|
|
|
|
x = x2;
|
|
y = y2;
|
|
style.symbol = LINE_NONE;
|
|
is_first = false;
|
|
}
|
|
}
|
|
|
|
x2 = b->x + xinc;
|
|
y2 = b->y - 1;
|
|
if (is_first && nth == 0 && x2 > x) {
|
|
xinc += 4;
|
|
x += 4;
|
|
}
|
|
r_cons_canvas_line (g->can, x, y, x2, y2, &style);
|
|
}
|
|
|
|
static void agraph_print_edges(const RAGraph *g) {
|
|
const RList *nodes = r_graph_get_nodes (g->graph);
|
|
RGraphNode *gn, *gv;
|
|
RListIter *it, *itn;
|
|
RANode *u, *v;
|
|
|
|
graph_foreach_anode (nodes, it, gn, u) {
|
|
const RList *neighbours = r_graph_get_neighbours (g->graph, gn);
|
|
const int exit_edges = r_list_length (neighbours);
|
|
int nth = 0;
|
|
|
|
graph_foreach_anode (neighbours, itn, gv, v) {
|
|
int cur_nth = nth;
|
|
if (g->is_callgraph) {
|
|
/* hack: we don't support more than two exit edges from a node
|
|
* yet, so set nth to zero, to make every edge appears as the
|
|
* "true" edge of the node */
|
|
cur_nth = 0;
|
|
} else if (exit_edges == 1) {
|
|
cur_nth = -1;
|
|
}
|
|
agraph_print_edge (g, u, v, cur_nth);
|
|
nth++;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void agraph_toggle_callgraph(RAGraph *g) {
|
|
g->is_callgraph = !g->is_callgraph;
|
|
g->need_reload_nodes = true;
|
|
g->force_update_seek = true;
|
|
}
|
|
|
|
static void agraph_set_zoom (RAGraph *g, int v) {
|
|
g->mode = v <= 0 ? R_AGRAPH_MODE_MINI : R_AGRAPH_MODE_NORMAL;
|
|
g->zoom = R_MAX (0, v);
|
|
g->need_update_dim = true;
|
|
g->need_set_layout = true;
|
|
}
|
|
|
|
/* reload all the info in the nodes, depending on the type of the graph
|
|
* (callgraph, CFG, etc.), set the default layout for these nodes and center
|
|
* the screen on the selected one */
|
|
static int agraph_reload_nodes(RAGraph *g, RCore *core, RAnalFunction *fcn) {
|
|
r_agraph_reset (g);
|
|
return reload_nodes (g, core, fcn);
|
|
}
|
|
|
|
static void follow_nth(RAGraph *g, int nth) {
|
|
const RGraphNode *cn = r_graph_nth_neighbour (g->graph, g->curnode, nth);
|
|
if (cn) r_agraph_set_curnode (g, get_anode (cn));
|
|
}
|
|
|
|
static void agraph_follow_true(RAGraph *g) {
|
|
follow_nth (g, 0);
|
|
agraph_update_seek (g, get_anode (g->curnode), false);
|
|
}
|
|
|
|
static void agraph_follow_false(RAGraph *g) {
|
|
follow_nth (g, 1);
|
|
agraph_update_seek (g, get_anode (g->curnode), false);
|
|
}
|
|
|
|
/* seek the next node in visual order */
|
|
static void agraph_next_node(RAGraph *g) {
|
|
r_agraph_set_curnode (g, get_anode (find_near_of (g, g->curnode, true)));
|
|
agraph_update_seek (g, get_anode (g->curnode), false);
|
|
}
|
|
|
|
/* seek the previous node in visual order */
|
|
static void agraph_prev_node(RAGraph *g) {
|
|
r_agraph_set_curnode (g, get_anode (find_near_of (g, g->curnode, false)));
|
|
agraph_update_seek (g, get_anode (g->curnode), false);
|
|
}
|
|
|
|
static void agraph_update_title (RAGraph *g, RAnalFunction *fcn) {
|
|
const char *mode_str = g->is_callgraph ? mode2str(g, "CG") : mode2str (g, "BB");
|
|
char *new_title = r_str_newf(
|
|
"[0x%08"PFMT64x"]> VV @ %s (nodes %d edges %d zoom %d%%) %s mouse:%s movements-speed:%d",
|
|
fcn->addr, fcn->name, g->graph->n_nodes, g->graph->n_edges,
|
|
g->zoom, mode_str, mousemodes[mousemode], g->movspeed);
|
|
r_agraph_set_title (g, new_title);
|
|
r_str_free (new_title);
|
|
}
|
|
|
|
/* look for any change in the state of the graph
|
|
* and update what's necessary */
|
|
static int check_changes(RAGraph *g, int is_interactive,
|
|
RCore *core, RAnalFunction *fcn) {
|
|
int oldpos[2] = {0, 0};
|
|
if (g->need_reload_nodes && core) {
|
|
if (!g->update_seek_on && !g->force_update_seek) {
|
|
// save scroll here
|
|
oldpos[0] = g->can->sx;
|
|
oldpos[1] = g->can->sy;
|
|
}
|
|
if (!agraph_reload_nodes (g, core, fcn)) {
|
|
return false;
|
|
}
|
|
}
|
|
if (fcn) {
|
|
agraph_update_title (g, fcn);
|
|
}
|
|
if (g->need_update_dim || g->need_reload_nodes || !is_interactive) {
|
|
update_node_dimension (g->graph, is_mini (g), g->zoom);
|
|
}
|
|
if (g->need_set_layout || g->need_reload_nodes || !is_interactive) {
|
|
agraph_set_layout (g, is_interactive);
|
|
}
|
|
if (core) {
|
|
ut64 off = r_core_anal_get_bbaddr (core, core->offset);
|
|
char *title = get_title (off);
|
|
RANode *cur_anode = get_anode (g->curnode);
|
|
if (fcn && ((is_interactive && !cur_anode) ||
|
|
(cur_anode && strcmp (cur_anode->title, title) != 0))) {
|
|
g->update_seek_on = r_agraph_get_node (g, title);
|
|
if (g->update_seek_on) {
|
|
r_agraph_set_curnode (g, g->update_seek_on);
|
|
g->force_update_seek = true;
|
|
}
|
|
}
|
|
free (title);
|
|
|
|
g->can->color = r_config_get_i (core->config, "scr.color");
|
|
}
|
|
if (g->update_seek_on || g->force_update_seek) {
|
|
RANode *n = g->update_seek_on;
|
|
if (!n && g->curnode) n = get_anode (g->curnode);
|
|
if (n) update_seek (g->can, n, g->force_update_seek);
|
|
}
|
|
if (oldpos[0] || oldpos[1]) {
|
|
g->can->sx = oldpos[0];
|
|
g->can->sy = oldpos[1];
|
|
}
|
|
g->need_reload_nodes = false;
|
|
g->need_update_dim = false;
|
|
g->need_set_layout = false;
|
|
g->update_seek_on = NULL;
|
|
g->force_update_seek = false;
|
|
return true;
|
|
}
|
|
|
|
static int agraph_print(RAGraph *g, int is_interactive, RCore *core, RAnalFunction *fcn) {
|
|
int h, w = r_cons_get_size (&h);
|
|
int ret = check_changes (g, is_interactive, core, fcn);
|
|
if (!ret) return false;
|
|
|
|
if (is_interactive) {
|
|
r_cons_clear00 ();
|
|
} else {
|
|
/* TODO: limit to screen size when the output is not redirected to file */
|
|
update_graph_sizes (g);
|
|
}
|
|
|
|
h = is_interactive ? h : g->h + 1;
|
|
w = is_interactive ? w : g->w;
|
|
r_cons_canvas_resize (g->can, w, h);
|
|
//r_cons_canvas_clear (g->can);
|
|
if (!is_interactive) {
|
|
g->can->sx = -g->x;
|
|
g->can->sy = -g->y + 1;
|
|
}
|
|
|
|
agraph_print_edges (g);
|
|
agraph_print_nodes (g);
|
|
|
|
/* print the graph title */
|
|
(void)G (-g->can->sx, -g->can->sy);
|
|
W (g->title);
|
|
if (is_interactive && g->title) {
|
|
int title_len = strlen (g->title);
|
|
r_cons_canvas_fill (g->can, -g->can->sx + title_len, -g->can->sy,
|
|
w - title_len, 1, ' ', true);
|
|
}
|
|
|
|
r_cons_canvas_print_region (g->can);
|
|
|
|
if (is_interactive) {
|
|
const char *cmdv;
|
|
cmdv = r_config_get (core->config, "cmd.gprompt");
|
|
if (cmdv && *cmdv) {
|
|
r_cons_gotoxy (0, 1);
|
|
r_core_cmd0 (core, cmdv);
|
|
}
|
|
r_cons_flush ();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int agraph_refresh(struct agraph_refresh_data *grd) {
|
|
RCore *core = grd->core;
|
|
RAGraph *g = grd->g;
|
|
RAnalFunction *f, **fcn = grd->fcn;
|
|
|
|
// allow to change the current function during debugging
|
|
if (g->is_instep && core->io->debug) {
|
|
// seek only when the graph node changes
|
|
const char *pc = r_reg_get_name (core->dbg->reg, R_REG_NAME_PC);
|
|
RRegItem *r = r_reg_get (core->dbg->reg, pc, -1);
|
|
ut64 addr = r_reg_get_value (core->dbg->reg, r);
|
|
RANode *acur = get_anode (g->curnode);
|
|
char *title;
|
|
|
|
addr = r_core_anal_get_bbaddr (core, addr);
|
|
title = get_title (addr);
|
|
|
|
if (!acur || strcmp (acur->title, title) != 0) {
|
|
r_core_cmd0 (core, "sr PC");
|
|
}
|
|
free (title);
|
|
g->is_instep = false;
|
|
}
|
|
|
|
if (fcn) {
|
|
f = r_anal_get_fcn_in (core->anal, core->offset, 0);
|
|
if (!f) {
|
|
r_cons_message ("Not in a function. Type 'df' to define it here");
|
|
return 0;
|
|
} else if (f && f != *fcn) {
|
|
*fcn = f;
|
|
g->need_reload_nodes = true;
|
|
g->force_update_seek = true;
|
|
}
|
|
}
|
|
|
|
return agraph_print (g, grd->fs, core, fcn != NULL ? *fcn : NULL);
|
|
}
|
|
|
|
static void agraph_toggle_speed (RAGraph *g, RCore *core) {
|
|
int alt = r_config_get_i (core->config, "graph.scroll");
|
|
g->movspeed = g->movspeed == DEFAULT_SPEED ? alt : DEFAULT_SPEED;
|
|
}
|
|
|
|
static void agraph_init(RAGraph *g) {
|
|
g->is_callgraph = false;
|
|
g->is_instep = false;
|
|
g->need_reload_nodes = true;
|
|
g->force_update_seek = true;
|
|
g->color_box = Color_RESET;
|
|
g->color_box2 = Color_BLUE; // selected node
|
|
g->color_box3 = Color_MAGENTA;
|
|
g->graph = r_graph_new ();
|
|
g->nodes = sdb_new0 ();
|
|
g->zoom = ZOOM_DEFAULT;
|
|
g->movspeed = DEFAULT_SPEED; //r_config_get_i (g->core->config, "graph.scroll");
|
|
g->db = sdb_new0 ();
|
|
}
|
|
|
|
static void free_anode (RANode *n) {
|
|
free (n->title);
|
|
free (n->body);
|
|
}
|
|
|
|
static int free_anode_cb (void *user UNUSED, const char *k UNUSED, const char *v) {
|
|
RANode *n = (RANode *)(size_t)sdb_atoi(v);
|
|
free_anode (n);
|
|
return 1;
|
|
}
|
|
|
|
static void agraph_free_nodes (const RAGraph *g) {
|
|
sdb_foreach (g->nodes, (SdbForeachCallback)free_anode_cb, NULL);
|
|
sdb_free (g->nodes);
|
|
}
|
|
|
|
static void sdb_set_enc (Sdb *db, const char *key, const char *v, ut32 cas) {
|
|
char *estr = sdb_encode ((const void *)v, -1);
|
|
sdb_set (db, key, estr, cas);
|
|
free (estr);
|
|
}
|
|
|
|
static void agraph_sdb_init (const RAGraph *g) {
|
|
sdb_bool_set (g->db, "agraph.is_callgraph", g->is_callgraph, 0);
|
|
|
|
sdb_set_enc (g->db, "agraph.color_box", g->color_box, 0);
|
|
sdb_set_enc (g->db, "agraph.color_box2", g->color_box2, 0);
|
|
sdb_set_enc (g->db, "agraph.color_box3", g->color_box3, 0);
|
|
sdb_set_enc (g->db, "agraph.color_true", g->color_true, 0);
|
|
sdb_set_enc (g->db, "agraph.color_false", g->color_false, 0);
|
|
}
|
|
|
|
R_API Sdb *r_agraph_get_sdb (RAGraph *g) {
|
|
g->need_update_dim = true;
|
|
g->need_set_layout = true;
|
|
check_changes (g, false, NULL, NULL);
|
|
return g->db;
|
|
}
|
|
|
|
R_API void r_agraph_print (RAGraph *g) {
|
|
agraph_print (g, false, NULL, NULL);
|
|
if (g->graph->n_nodes > 0)
|
|
r_cons_newline ();
|
|
}
|
|
|
|
R_API void r_agraph_set_title (RAGraph *g, const char *title) {
|
|
free (g->title);
|
|
g->title = title ? strdup (title) : NULL;
|
|
sdb_set (g->db, "agraph.title", g->title, 0);
|
|
}
|
|
|
|
R_API RANode *r_agraph_add_node(const RAGraph *g, const char *title, const char *body) {
|
|
RANode *res = r_agraph_get_node (g, title);
|
|
if (res) return res;
|
|
res = R_NEW0 (RANode);
|
|
if (!res) return NULL;
|
|
res->title = title ? strdup(title) : strdup ("");
|
|
res->body = body ? strdup(body) : strdup ("");
|
|
res->layer = -1;
|
|
res->pos_in_layer = -1;
|
|
res->is_dummy = false;
|
|
res->is_reversed = false;
|
|
res->klass = -1;
|
|
res->gnode = r_graph_add_node (g->graph, res);
|
|
sdb_num_set (g->nodes, title, (ut64)(size_t)res, 0);
|
|
if (res->title) {
|
|
char *s, *estr, *b;
|
|
size_t len;
|
|
|
|
sdb_array_add (g->db, "agraph.nodes", res->title, 0);
|
|
b = strdup (res->body);
|
|
len = strlen (b);
|
|
if (len>0 && b[len-1] == '\n')
|
|
b[len - 1] = '\0';
|
|
estr = sdb_encode ((const void *)b, -1);
|
|
s = sdb_fmt (1, "base64:%s", estr);
|
|
free (estr);
|
|
free (b);
|
|
sdb_set (g->db, sdb_fmt (2, "agraph.nodes.%s.body", res->title), s, 0);
|
|
}
|
|
return res;
|
|
}
|
|
|
|
R_API bool r_agraph_del_node(const RAGraph *g, const char *title) {
|
|
RANode *an, *res = r_agraph_get_node (g, title);
|
|
const RList *innodes;
|
|
RGraphNode *gn;
|
|
RListIter *it;
|
|
|
|
if (!res) return false;
|
|
sdb_set (g->nodes, title, NULL, 0);
|
|
sdb_array_remove (g->db, "agraph.nodes", res->title, 0);
|
|
sdb_set (g->db, sdb_fmt (2, "agraph.nodes.%s", res->title), NULL, 0);
|
|
sdb_set (g->db, sdb_fmt (2, "agraph.nodes.%s.body", res->title), 0, 0);
|
|
sdb_set (g->db, sdb_fmt (2, "agraph.nodes.%s.x", res->title), NULL, 0);
|
|
sdb_set (g->db, sdb_fmt (2, "agraph.nodes.%s.y", res->title), NULL, 0);
|
|
sdb_set (g->db, sdb_fmt (2, "agraph.nodes.%s.w", res->title), NULL, 0);
|
|
sdb_set (g->db, sdb_fmt (2, "agraph.nodes.%s.h", res->title), NULL, 0);
|
|
sdb_set (g->db, sdb_fmt (2, "agraph.nodes.%s.neighbours", res->title),
|
|
NULL, 0);
|
|
|
|
innodes = r_graph_innodes (g->graph, res->gnode);
|
|
graph_foreach_anode (innodes, it, gn, an) {
|
|
const char *key = sdb_fmt (2, "agraph.nodes.%s.neighbours", an->title);
|
|
sdb_array_remove (g->db, key, res->title, 0);
|
|
}
|
|
|
|
r_graph_del_node (g->graph, res->gnode);
|
|
res->gnode = NULL;
|
|
|
|
free_anode (res);
|
|
return true;
|
|
}
|
|
|
|
static int user_node_cb(struct g_cb *user, const char *k UNUSED, const char *v) {
|
|
RANodeCallback cb = user->node_cb;
|
|
void *user_data = user->data;
|
|
RANode *n = (RANode *)(size_t)sdb_atoi (v);
|
|
if (n) cb (n, user_data);
|
|
return 1;
|
|
}
|
|
|
|
static int user_edge_cb(struct g_cb *user, const char *k UNUSED, const char *v) {
|
|
RAEdgeCallback cb = user->edge_cb;
|
|
RAGraph *g = user->graph;
|
|
void *user_data = user->data;
|
|
RANode *an, *n = (RANode *)(size_t)sdb_atoi (v);
|
|
const RList *neigh = r_graph_get_neighbours (g->graph, n->gnode);
|
|
RListIter *it;
|
|
RGraphNode *gn;
|
|
|
|
graph_foreach_anode (neigh, it, gn, an) {
|
|
cb (n, an, user_data);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
R_API void r_agraph_foreach(RAGraph *g, RANodeCallback cb, void *user) {
|
|
struct g_cb u;
|
|
u.node_cb = cb;
|
|
u.data = user;
|
|
sdb_foreach (g->nodes, (SdbForeachCallback)user_node_cb, &u);
|
|
}
|
|
|
|
R_API void r_agraph_foreach_edge(RAGraph *g, RAEdgeCallback cb, void *user) {
|
|
struct g_cb u;
|
|
u.graph = g;
|
|
u.edge_cb = cb;
|
|
u.data = user;
|
|
sdb_foreach (g->nodes, (SdbForeachCallback)user_edge_cb, &u);
|
|
}
|
|
|
|
R_API RANode *r_agraph_get_first_node(const RAGraph *g) {
|
|
const RList *l = r_graph_get_nodes (g->graph);
|
|
RGraphNode *rgn = r_list_first (l);
|
|
return get_anode (rgn);
|
|
}
|
|
|
|
R_API RANode *r_agraph_get_node(const RAGraph *g, const char *title) {
|
|
return (RANode *)(size_t)sdb_num_get (g->nodes, title, NULL);
|
|
}
|
|
|
|
R_API void r_agraph_add_edge(const RAGraph *g, RANode *a, RANode *b) {
|
|
if (!g || !a || !b) return;
|
|
r_graph_add_edge (g->graph, a->gnode, b->gnode);
|
|
if (a->title && b->title) {
|
|
char *k = sdb_fmt (1, "agraph.nodes.%s.neighbours", a->title);
|
|
sdb_array_add (g->db, k, b->title, 0);
|
|
}
|
|
}
|
|
|
|
R_API void r_agraph_add_edge_at(const RAGraph *g, RANode *a, RANode *b, int nth) {
|
|
if (!g || !a || !b) return;
|
|
if (a->title && b->title) {
|
|
char *k = sdb_fmt (1, "agraph.nodes.%s.neighbours", a->title);
|
|
sdb_array_insert (g->db, k, nth, b->title, 0);
|
|
}
|
|
r_graph_add_edge_at (g->graph, a->gnode, b->gnode, nth);
|
|
}
|
|
|
|
R_API void r_agraph_del_edge(const RAGraph *g, RANode *a, RANode *b) {
|
|
if (!g || !a || !b) return;
|
|
if (a->title && b->title) {
|
|
char *k = sdb_fmt (1, "agraph.nodes.%s.neighbours", a->title);
|
|
sdb_array_remove (g->db, k, b->title, 0);
|
|
}
|
|
r_graph_del_edge (g->graph, a->gnode, b->gnode);
|
|
}
|
|
|
|
R_API void r_agraph_reset (RAGraph *g) {
|
|
r_graph_reset (g->graph);
|
|
agraph_free_nodes (g);
|
|
r_agraph_set_title (g, NULL);
|
|
sdb_reset (g->db);
|
|
r_list_free (g->edges);
|
|
|
|
g->nodes = sdb_new0 ();
|
|
g->update_seek_on = NULL;
|
|
g->x = g->y = g->w = g->h = 0;
|
|
agraph_sdb_init (g);
|
|
g->edges = r_list_new ();
|
|
g->curnode = NULL;
|
|
}
|
|
|
|
R_API void r_agraph_free(RAGraph *g) {
|
|
r_graph_free (g->graph);
|
|
if (g->edges) r_list_free (g->edges);
|
|
agraph_free_nodes (g);
|
|
r_agraph_set_title (g, NULL);
|
|
sdb_free (g->db);
|
|
free (g);
|
|
}
|
|
|
|
R_API RAGraph *r_agraph_new(RConsCanvas *can) {
|
|
RAGraph *g = R_NEW0 (RAGraph);
|
|
if (!g) return NULL;
|
|
g->can = can;
|
|
agraph_init (g);
|
|
agraph_sdb_init (g);
|
|
return g;
|
|
}
|
|
|
|
static void visual_offset (RAGraph *g, RCore *core) {
|
|
char buf[256];
|
|
int rows;
|
|
r_cons_get_size (&rows);
|
|
r_cons_gotoxy (0, rows);
|
|
r_cons_flush ();
|
|
r_line_set_prompt ("[offset]> ");
|
|
strcpy (buf, "s ");
|
|
if (r_cons_fgets (buf + 2, sizeof (buf) - 3, 0, NULL) > 0) {
|
|
if (buf[2] == '.') {
|
|
buf[1] = '.';
|
|
}
|
|
r_core_cmd0 (core, buf);
|
|
}
|
|
}
|
|
|
|
static void goto_asmqjmps(RAGraph *g, RCore *core) {
|
|
const char *h = "[Fast goto call/jmp]> ";
|
|
char obuf[R_CORE_ASMQJMPS_LEN_LETTERS + 1];
|
|
int rows, i = 0;
|
|
bool cont;
|
|
|
|
r_cons_get_size (&rows);
|
|
r_cons_gotoxy (0, rows);
|
|
r_cons_printf (Color_RESET);
|
|
r_cons_printf (h);
|
|
r_cons_flush ();
|
|
r_cons_clear_line (0);
|
|
r_cons_gotoxy (strlen (h) + 1, rows);
|
|
|
|
do {
|
|
char ch = r_cons_readchar ();
|
|
obuf[i++] = ch;
|
|
r_cons_printf ("%c", ch);
|
|
r_cons_flush ();
|
|
cont = isalpha ((ut8)ch) && !islower ((ut8)ch);
|
|
} while (i < R_CORE_ASMQJMPS_LEN_LETTERS && cont);
|
|
|
|
obuf[i] = '\0';
|
|
ut64 addr = r_core_get_asmqjmps (core, obuf);
|
|
if (addr != UT64_MAX) {
|
|
char *title = get_title (addr);
|
|
RANode *addr_node = r_agraph_get_node (g, title);
|
|
if (addr_node) {
|
|
r_agraph_set_curnode (g, addr_node);
|
|
agraph_update_seek (g, addr_node, true);
|
|
} else {
|
|
r_io_sundo_push (core->io, core->offset);
|
|
r_core_seek (core, addr, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void seek_to_node(RANode *n, RCore *core) {
|
|
ut64 off = r_core_anal_get_bbaddr (core, core->offset);
|
|
char *title = get_title (off);
|
|
|
|
if (strcmp (title, n->title) != 0) {
|
|
char *cmd = r_str_newf ("s %s", n->title);
|
|
r_core_cmd0 (core, cmd);
|
|
free (cmd);
|
|
}
|
|
free (title);
|
|
}
|
|
|
|
R_API int r_core_visual_graph(RCore *core, RAGraph *g, RAnalFunction *_fcn, int is_interactive) {
|
|
int o_asmqjmps_letter = core->is_asmqjmps_letter;
|
|
int o_scrinteractive = r_config_get_i (core->config, "scr.interactive");
|
|
int o_vmode = core->vmode;
|
|
int exit_graph = false, is_error = false;
|
|
struct agraph_refresh_data *grd;
|
|
int okey, key, wheel;
|
|
RAnalFunction *fcn = NULL;
|
|
const char *key_s;
|
|
RConsCanvas *can, *o_can = NULL;
|
|
bool graph_allocated = false;
|
|
int movspeed;
|
|
int w, h;
|
|
int ret;
|
|
int invscroll;
|
|
|
|
w = r_cons_get_size (&h);
|
|
can = r_cons_canvas_new (w, h);
|
|
if (!can) {
|
|
eprintf ("Cannot create RCons.canvas context\n");
|
|
return false;
|
|
}
|
|
can->linemode = 1;
|
|
can->color = r_config_get_i (core->config, "scr.color");
|
|
r_config_set_i (core->config, "scr.interactive", false);
|
|
|
|
if (!g) {
|
|
graph_allocated = true;
|
|
fcn = _fcn ? _fcn : r_anal_get_fcn_in (core->anal, core->offset, 0);
|
|
if (!fcn) {
|
|
eprintf ("No function in current seek\n");
|
|
return false;
|
|
}
|
|
g = r_agraph_new (can);
|
|
if (!g) {
|
|
is_error = true;
|
|
goto err_graph_new;
|
|
}
|
|
} else {
|
|
o_can = g->can;
|
|
}
|
|
g->can = can;
|
|
g->movspeed = r_config_get_i (core->config, "graph.scroll");
|
|
g->on_curnode_change = (RANodeCallback)seek_to_node;
|
|
g->on_curnode_change_data = core;
|
|
int asm_comments = r_config_get_i (core->config, "asm.comments");
|
|
r_config_set (core->config, "asm.comments", r_str_bool (false));
|
|
|
|
/* we want letters as shortcuts for call/jmps */
|
|
core->is_asmqjmps_letter = true;
|
|
core->vmode = true;
|
|
|
|
grd = R_NEW0 (struct agraph_refresh_data);
|
|
grd->g = g;
|
|
grd->fs = is_interactive;
|
|
grd->core = core;
|
|
grd->fcn = fcn != NULL ? &fcn : NULL;
|
|
ret = agraph_refresh (grd);
|
|
if (!ret || !is_interactive) {
|
|
r_cons_newline ();
|
|
exit_graph = true;
|
|
is_error = !ret;
|
|
}
|
|
|
|
core->cons->event_data = grd;
|
|
core->cons->event_resize = (RConsEvent)agraph_refresh;
|
|
|
|
while (!exit_graph && !is_error) {
|
|
invscroll = r_config_get_i (core->config, "graph.invscroll");
|
|
ret = agraph_refresh (grd);
|
|
if (!ret) {
|
|
is_error = true;
|
|
break;
|
|
}
|
|
|
|
r_cons_show_cursor (false);
|
|
wheel = r_config_get_i (core->config, "scr.wheel");
|
|
if (wheel) {
|
|
r_cons_enable_mouse (true);
|
|
}
|
|
|
|
// r_core_graph_inputhandle()
|
|
okey = r_cons_readchar ();
|
|
key = r_cons_arrow_to_hjkl (okey);
|
|
|
|
if (core->cons->mouse_event) {
|
|
movspeed = r_config_get_i (core->config, "scr.wheelspeed");
|
|
switch (key) {
|
|
case 'j':
|
|
case 'k':
|
|
switch (mousemode) {
|
|
case 0: break;
|
|
case 1: key = key=='k'?'h':'l'; break;
|
|
case 2: key = key=='k'?'J':'K'; break;
|
|
case 3: key = key=='k'?'L':'H'; break;
|
|
}
|
|
break;
|
|
}
|
|
} else {
|
|
movspeed = g->movspeed;
|
|
}
|
|
|
|
switch (key) {
|
|
case '-':
|
|
agraph_set_zoom (g, g->zoom - ZOOM_STEP);
|
|
break;
|
|
case '+':
|
|
agraph_set_zoom (g, g->zoom + ZOOM_STEP);
|
|
break;
|
|
case '0':
|
|
agraph_set_zoom (g, ZOOM_DEFAULT);
|
|
agraph_update_seek (g, get_anode (g->curnode), true);
|
|
break;
|
|
case '|':
|
|
{ // TODO: edit
|
|
const char *buf = NULL;
|
|
const char *cmd = r_config_get (core->config, "cmd.gprompt");
|
|
r_line_set_prompt ("cmd.gprompt> ");
|
|
core->cons->line->contents = strdup (cmd);
|
|
buf = r_line_readline ();
|
|
core->cons->line->contents = NULL;
|
|
r_config_set (core->config, "cmd.gprompt", buf);
|
|
}
|
|
break;
|
|
case 'V':
|
|
if (fcn) agraph_toggle_callgraph(g);
|
|
break;
|
|
case 'Z':
|
|
if (okey == 27) agraph_prev_node (g);
|
|
break;
|
|
case 's':
|
|
key_s = r_config_get (core->config, "key.s");
|
|
if (key_s && *key_s) {
|
|
r_core_cmd0 (core, key_s);
|
|
} else {
|
|
if (r_config_get_i (core->config, "cfg.debug"))
|
|
r_core_cmd0 (core, "ds;.dr*");
|
|
else
|
|
r_core_cmd0 (core, "aes;.dr*");
|
|
}
|
|
g->is_instep = true;
|
|
g->need_reload_nodes = true;
|
|
break;
|
|
case 'S':
|
|
if (r_config_get_i (core->config, "cfg.debug")) {
|
|
r_core_cmd0 (core, "dso;.dr*");
|
|
} else {
|
|
r_core_cmd0 (core, "aeso;.dr*");
|
|
}
|
|
g->is_instep = true;
|
|
g->need_reload_nodes = true;
|
|
break;
|
|
case 'x':
|
|
case 'X':
|
|
{
|
|
if (!fcn) break;
|
|
|
|
ut64 old_off = core->offset;
|
|
ut64 off = r_core_anal_get_bbaddr (core, core->offset);
|
|
r_core_seek (core, off, 0);
|
|
if ((key == 'x' && !r_core_visual_xrefs_x (core)) ||
|
|
(key == 'X' && !r_core_visual_xrefs_X (core))) {
|
|
r_core_seek(core, old_off, 0);
|
|
}
|
|
break;
|
|
}
|
|
case 9: // tab
|
|
agraph_next_node (g);
|
|
break;
|
|
case '?':
|
|
r_cons_clear00 ();
|
|
r_cons_printf ("Visual Ascii Art graph keybindings:\n"
|
|
" . - center graph to the current node\n"
|
|
" :cmd - run radare command\n"
|
|
" ' - toggle asm.comments\n"
|
|
" ; - add comment in current basic block\n"
|
|
" / - highlight text\n"
|
|
" Page-UP/DOWN - go to the top/bottom of the canvas\n"
|
|
" C - toggle scr.colors\n"
|
|
" hjkl - scroll canvas\n"
|
|
" HJKL - move node\n"
|
|
" tab - select next node\n"
|
|
" TAB - select previous node\n"
|
|
" t/f - follow true/false edges\n"
|
|
" g([A-Za-z]*) - follow jmp/call identified by shortcut\n"
|
|
" r - refresh graph\n"
|
|
" R - randomize colors\n"
|
|
" o - go/seek to given offset\n"
|
|
" u/U - undo/redo seek\n"
|
|
" p/P - rotate graph modes (normal, display offsets, esil, minigraph)\n"
|
|
" s/S - step / step over\n"
|
|
" V - toggle basicblock / call graphs\n"
|
|
" w - toggle between movements speed 1 and graph.scroll\n"
|
|
" x/X - jump to xref/ref\n"
|
|
" +/-/0 - zoom in/out/default\n");
|
|
r_cons_flush ();
|
|
r_cons_any_key (NULL);
|
|
break;
|
|
case 'p':
|
|
if (!fcn) break;
|
|
g->mode = next_mode (g->mode);
|
|
g->need_reload_nodes = true;
|
|
break;
|
|
case 'P':
|
|
if (!fcn) break;
|
|
g->mode = prev_mode (g->mode);
|
|
g->need_reload_nodes = true;
|
|
break;
|
|
case 'g':
|
|
goto_asmqjmps (g, core);
|
|
break;
|
|
case 'o':
|
|
visual_offset (g, core);
|
|
break;
|
|
case 'u':
|
|
{
|
|
if (!fcn) break;
|
|
ut64 off = r_io_sundo (core->io, core->offset);
|
|
if (off != UT64_MAX) r_core_seek (core, off, 0);
|
|
else eprintf ("Can not undo\n");
|
|
break;
|
|
}
|
|
case 'U':
|
|
{
|
|
if (!fcn) break;
|
|
ut64 off = r_io_sundo_redo (core->io);
|
|
if (off != UT64_MAX) r_core_seek (core, off, 0);
|
|
else eprintf ("Cannot redo\n");
|
|
break;
|
|
}
|
|
case 'R':
|
|
if (!fcn) break;
|
|
r_core_cmd0 (core, "ecr");
|
|
g->color_box = core->cons->pal.graph_box;
|
|
g->color_box2 = core->cons->pal.graph_box2;
|
|
g->color_box3 = core->cons->pal.graph_box3;
|
|
g->color_true = core->cons->pal.graph_true;
|
|
g->color_false = core->cons->pal.graph_false;
|
|
get_bbupdate (g, core, fcn);
|
|
break;
|
|
case '!':
|
|
r_core_visual_panels (core);
|
|
break;
|
|
case '\'':
|
|
if (!fcn) break;
|
|
r_config_toggle (core->config, "asm.comments");
|
|
g->need_reload_nodes = true;
|
|
break;
|
|
case ';':
|
|
{
|
|
if (!fcn) break;
|
|
char buf[256];
|
|
r_line_set_prompt ("[comment]> ");
|
|
if (r_cons_fgets (buf, sizeof (buf) - 1, 0, NULL) > 0) {
|
|
r_core_cmdf (core, "\"CC %s\"", buf);
|
|
}
|
|
g->need_reload_nodes = true;
|
|
}
|
|
break;
|
|
case 'C':
|
|
r_config_toggle (core->config, "scr.color");
|
|
break;
|
|
case 'r':
|
|
if (!fcn) break;
|
|
g->need_reload_nodes = true;
|
|
break;
|
|
case 'm':
|
|
mousemode++;
|
|
if (!mousemodes[mousemode])
|
|
mousemode = 0;
|
|
break;
|
|
case 'M':
|
|
mousemode--;
|
|
if (mousemode<0)
|
|
mousemode = 3;
|
|
break;
|
|
case 'J':
|
|
if (okey == 27 && r_cons_readchar () == 126) {
|
|
// handle page down key
|
|
can->sy -= PAGEKEY_SPEED * (invscroll ? -1 : 1);
|
|
} else {
|
|
get_anode(g->curnode)->y += movspeed;
|
|
}
|
|
break;
|
|
case 'K':
|
|
if (okey == 27 && r_cons_readchar () == 126) {
|
|
// handle page up key
|
|
can->sy += PAGEKEY_SPEED * (invscroll ? -1 : 1);
|
|
} else {
|
|
get_anode(g->curnode)->y -= movspeed;
|
|
}
|
|
break;
|
|
case 'F':
|
|
if (okey == 27) {
|
|
// handle end key
|
|
const RGraphNode *gn = find_near_of (g, NULL, false);
|
|
g->update_seek_on = get_anode (gn);
|
|
}
|
|
break;
|
|
case 'H':
|
|
if (okey == 27) {
|
|
// handle home key
|
|
const RGraphNode *gn = find_near_of (g, NULL, true);
|
|
g->update_seek_on = get_anode (gn);
|
|
} else {
|
|
get_anode(g->curnode)->x -= movspeed;
|
|
}
|
|
break;
|
|
case 'L': get_anode(g->curnode)->x += movspeed; break;
|
|
case 'j': can->sy -= movspeed * (invscroll ? -1 : 1); break;
|
|
case 'k': can->sy += movspeed * (invscroll ? -1 : 1); break;
|
|
case 'l': can->sx -= movspeed * (invscroll ? -1 : 1); break;
|
|
case 'h': can->sx += movspeed * (invscroll ? -1 : 1); break;
|
|
case '.':
|
|
agraph_update_seek (g, get_anode (g->curnode), true);
|
|
break;
|
|
case 't':
|
|
agraph_follow_true (g);
|
|
break;
|
|
case 'f':
|
|
agraph_follow_false (g);
|
|
break;
|
|
case '/':
|
|
r_config_set_i (core->config, "scr.interactive", true);
|
|
r_core_cmd0 (core, "?i highlight;e scr.highlight=`?y`");
|
|
r_config_set_i (core->config, "scr.interactive", false);
|
|
break;
|
|
case ':':
|
|
r_core_visual_prompt_input (core);
|
|
get_bbupdate (g, core, fcn);
|
|
break;
|
|
case 'w':
|
|
agraph_toggle_speed (g, core);
|
|
break;
|
|
case -1: // EOF
|
|
case ' ':
|
|
case 'q':
|
|
if (g->is_callgraph) {
|
|
agraph_toggle_callgraph(g);
|
|
} else {
|
|
exit_graph = true;
|
|
}
|
|
break;
|
|
case 27: // ESC
|
|
if (r_cons_readchar () == 91) {
|
|
if (r_cons_readchar () == 90) {
|
|
agraph_prev_node (g);
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
r_config_set (core->config, "asm.comments", r_str_bool (asm_comments));
|
|
core->cons->event_data = NULL;
|
|
core->cons->event_resize = NULL;
|
|
core->vmode = o_vmode;
|
|
core->is_asmqjmps_letter = o_asmqjmps_letter;
|
|
core->keep_asmqjmps = false;
|
|
|
|
free (grd);
|
|
if (graph_allocated) {
|
|
r_agraph_free (g);
|
|
err_graph_new:
|
|
r_config_set_i (core->config, "scr.interactive", o_scrinteractive);
|
|
r_cons_canvas_free (can);
|
|
} else {
|
|
g->can = o_can;
|
|
}
|
|
return !is_error;
|
|
}
|