2013-01-16 12:17:14 +01:00
|
|
|
/* radare - LGPL - Copyright 2007-2013 - pancake */
|
2009-02-05 22:08:46 +01:00
|
|
|
|
|
|
|
#include <r_flags.h>
|
2009-06-15 02:44:05 +00:00
|
|
|
#include <r_util.h>
|
2010-02-21 11:35:49 +01:00
|
|
|
#include <r_cons.h>
|
2009-02-05 22:08:46 +01:00
|
|
|
#include <stdio.h>
|
2009-05-28 10:57:30 +00:00
|
|
|
|
2013-06-15 02:56:25 +02:00
|
|
|
R_LIB_VERSION(r_flag);
|
|
|
|
|
2010-03-08 12:45:22 +01:00
|
|
|
R_API RFlag * r_flag_new() {
|
2010-05-20 17:40:58 +02:00
|
|
|
int i;
|
2011-03-18 09:35:02 +01:00
|
|
|
RFlag *f = R_NEW (RFlag);
|
2011-05-21 21:05:21 +02:00
|
|
|
if (!f) return NULL;
|
2012-11-14 03:25:32 +01:00
|
|
|
f->base = 0;
|
2011-05-21 21:05:21 +02:00
|
|
|
f->flags = r_list_new ();
|
2012-11-14 03:25:32 +01:00
|
|
|
f->flags->free = (RListFree) r_flag_item_free;
|
2011-05-21 21:05:21 +02:00
|
|
|
f->space_idx = -1;
|
|
|
|
f->space_idx2 = -1;
|
2012-11-14 03:25:32 +01:00
|
|
|
f->ht_name = r_hashtable64_new ();
|
|
|
|
f->ht_off = r_hashtable64_new ();
|
2011-05-21 21:05:21 +02:00
|
|
|
for (i=0; i<R_FLAG_SPACES_MAX; i++)
|
|
|
|
f->spaces[i] = NULL;
|
2010-01-12 02:25:06 +01:00
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
2012-11-14 03:25:32 +01:00
|
|
|
R_API void r_flag_item_free (RFlagItem *item) {
|
|
|
|
free (item->cmd);
|
|
|
|
free (item->comment);
|
|
|
|
item->cmd = item->comment = NULL;
|
|
|
|
free (item);
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API RFlag *r_flag_free(RFlag *f) {
|
|
|
|
RFlagItem *item;
|
|
|
|
RListIter *iter;
|
|
|
|
r_list_foreach (f->flags, iter, item) {
|
|
|
|
RList *list = r_hashtable64_lookup (f->ht_name, item->namehash);
|
2014-03-02 04:36:27 +01:00
|
|
|
// XXX r_list_free (list);
|
2012-11-14 03:25:32 +01:00
|
|
|
list = r_hashtable64_lookup (f->ht_off, item->offset);
|
|
|
|
// XXX: segfault sometimes wtf -- r_list_free (list);
|
|
|
|
}
|
|
|
|
r_hashtable64_free (f->ht_off);
|
|
|
|
r_hashtable64_free (f->ht_name);
|
|
|
|
r_list_free (f->flags);
|
|
|
|
free (f);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-04-09 17:13:35 +02:00
|
|
|
R_API void r_flag_list(RFlag *f, int rad) {
|
2010-03-31 01:06:26 +02:00
|
|
|
int fs = -1;
|
2010-12-24 00:51:01 +01:00
|
|
|
RListIter *iter;
|
|
|
|
RFlagItem *flag;
|
|
|
|
|
2013-01-16 12:17:14 +01:00
|
|
|
if (rad=='j') {
|
|
|
|
int first = 1;
|
|
|
|
r_cons_printf ("[");
|
|
|
|
r_list_foreach_prev (f->flags, iter, flag) {
|
|
|
|
if ((f->space_idx != -1) && (flag->space != f->space_idx))
|
|
|
|
continue;
|
|
|
|
r_cons_printf ("%s{\"name\":\"%s\",\"size\":\"%"PFMT64d"\",\"offset\":%"PFMT64d,
|
|
|
|
first?"":",", flag->name, flag->size, flag->offset);
|
|
|
|
if (flag->comment)
|
|
|
|
r_cons_printf (",\"comment\":\"}");
|
|
|
|
else r_cons_printf ("}");
|
|
|
|
first = 0;
|
|
|
|
}
|
|
|
|
r_cons_printf ("]\n");
|
|
|
|
return;
|
|
|
|
}
|
2010-12-24 00:51:01 +01:00
|
|
|
r_list_foreach_prev (f->flags, iter, flag) {
|
2010-03-19 02:49:30 +01:00
|
|
|
if ((f->space_idx != -1) && (flag->space != f->space_idx))
|
|
|
|
continue;
|
2010-03-31 01:06:26 +02:00
|
|
|
if (rad) {
|
|
|
|
if (fs == -1 || flag->space != fs) {
|
2013-12-28 02:34:15 +01:00
|
|
|
const char *flagspace;
|
2010-03-31 01:06:26 +02:00
|
|
|
fs = flag->space;
|
2013-12-28 02:34:15 +01:00
|
|
|
flagspace = r_flag_space_get_i (f, fs);
|
|
|
|
if (!flagspace || !*flagspace)
|
|
|
|
flagspace = "*";
|
|
|
|
r_cons_printf ("fs %s\n", flagspace);
|
2010-03-31 01:06:26 +02:00
|
|
|
}
|
2012-11-14 03:25:32 +01:00
|
|
|
r_cons_printf ("f %s %"PFMT64d" 0x%08"PFMT64x" %s\n",
|
|
|
|
flag->name, flag->size, flag->offset,
|
|
|
|
flag->comment? flag->comment:"");
|
2013-07-04 03:34:28 +02:00
|
|
|
} else r_cons_printf ("0x%08"PFMT64x" %"PFMT64d" %s\n",
|
2010-03-19 02:49:30 +01:00
|
|
|
flag->offset, flag->size, flag->name);
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-08 12:45:22 +01:00
|
|
|
R_API RFlagItem *r_flag_get(RFlag *f, const char *name) {
|
2012-11-14 03:25:32 +01:00
|
|
|
RList *list = r_hashtable64_lookup (f->ht_name, r_str_hash64 (name));
|
|
|
|
if (list) {
|
|
|
|
RFlagItem *item = r_list_get_top (list);
|
|
|
|
return item;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-11-22 01:42:16 +01:00
|
|
|
|
2013-09-03 22:45:43 +02:00
|
|
|
R_API RFlagItem *r_flag_get_i2(RFlag *f, ut64 off) {
|
2014-02-13 03:11:23 +01:00
|
|
|
// TODO: this is buggy, do not use, must rewrite in sdb
|
2013-09-03 22:45:43 +02:00
|
|
|
RFlagItem *oitem = NULL;
|
|
|
|
RFlagItem *item = NULL;
|
|
|
|
RList *list = r_hashtable64_lookup (f->ht_off, off);
|
|
|
|
if (list) {
|
|
|
|
RListIter *iter;
|
|
|
|
r_list_foreach (list, iter, item) {
|
|
|
|
// XXX: hack, because some times the hashtable is poluted by ghost values
|
|
|
|
if (item->offset != off)
|
|
|
|
continue;
|
|
|
|
if (!strchr (item->name, '.'))
|
|
|
|
oitem = item;
|
|
|
|
if (strlen (item->name) < 5 || item->name[3]!='.')
|
|
|
|
continue;
|
|
|
|
oitem = item;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return oitem;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define R_FLAG_TEST 0
|
2010-03-08 12:45:22 +01:00
|
|
|
R_API RFlagItem *r_flag_get_i(RFlag *f, ut64 off) {
|
2012-11-14 03:25:32 +01:00
|
|
|
RList *list = r_hashtable64_lookup (f->ht_off, off);
|
|
|
|
if (list) {
|
|
|
|
RFlagItem *item = r_list_get_top (list);
|
|
|
|
#if R_FLAG_TEST
|
|
|
|
return item;
|
2009-05-28 10:57:30 +00:00
|
|
|
#else
|
2012-11-14 03:25:32 +01:00
|
|
|
// XXX: hack, because some times the hashtable is poluted by ghost values
|
|
|
|
if (item->offset == off)
|
|
|
|
return item;
|
|
|
|
#endif
|
2009-04-18 21:49:17 +00:00
|
|
|
}
|
|
|
|
return NULL;
|
2009-04-03 11:11:17 +00:00
|
|
|
}
|
|
|
|
|
2012-11-14 03:25:32 +01:00
|
|
|
R_API int r_flag_set(RFlag *f, const char *name, ut64 off, ut32 size, int dup) {
|
2013-04-16 01:48:03 +02:00
|
|
|
RFlagItem *item;
|
2012-11-14 03:25:32 +01:00
|
|
|
RList *list2, *list;
|
|
|
|
dup = 0; // XXX: force nondup
|
2010-06-21 01:58:45 +02:00
|
|
|
|
2013-04-16 01:48:03 +02:00
|
|
|
/* contract fail */
|
|
|
|
if (!name || !*name)
|
2013-03-04 00:33:14 +01:00
|
|
|
return R_FALSE;
|
2012-11-14 03:25:32 +01:00
|
|
|
if (dup) {
|
2013-09-03 22:45:43 +02:00
|
|
|
// XXX: doesnt works well
|
2013-04-16 01:48:03 +02:00
|
|
|
item = R_NEW0 (RFlagItem);
|
|
|
|
if (!r_flag_item_set_name (item, name)) {
|
|
|
|
eprintf ("Invalid flag name '%s'.\n", name);
|
|
|
|
free (item);
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
2012-11-14 03:25:32 +01:00
|
|
|
item->space = f->space_idx;
|
|
|
|
r_list_append (f->flags, item);
|
|
|
|
|
|
|
|
item->offset = off + f->base;
|
|
|
|
item->size = size;
|
|
|
|
|
|
|
|
list = r_hashtable64_lookup (f->ht_name, item->namehash);
|
|
|
|
if (!list) {
|
|
|
|
list = r_list_new ();
|
|
|
|
r_hashtable64_insert (f->ht_name, item->namehash, list);
|
|
|
|
}
|
|
|
|
r_list_append (list, item);
|
|
|
|
|
|
|
|
list2 = r_hashtable64_lookup (f->ht_off, off);
|
|
|
|
if (list2 == NULL) {
|
|
|
|
list2 = r_list_new ();
|
|
|
|
r_hashtable64_insert (f->ht_name, off, list2);
|
|
|
|
}
|
|
|
|
r_list_append (list2, item);
|
|
|
|
} else {
|
|
|
|
RListIter *iter2 = NULL;
|
|
|
|
RFlagItem *item2 = NULL, *item = r_flag_get (f, name);
|
|
|
|
if (item) {
|
|
|
|
RList *list2, *lol;
|
|
|
|
if (item->offset == off)
|
|
|
|
return R_TRUE;
|
|
|
|
/* remove old entry */
|
|
|
|
list2 = r_hashtable64_lookup (f->ht_off, item->offset);
|
|
|
|
if (list2)
|
|
|
|
/* No _safe loop necessary because we break immediately after the delete. */
|
|
|
|
r_list_foreach (list2, iter2, item2) {
|
|
|
|
if (item->namehash != item2->namehash)
|
|
|
|
continue;
|
|
|
|
if (item2->offset == item->offset) {
|
|
|
|
// r_list_delete (list2, iter2);
|
|
|
|
// delete without freeing contents
|
|
|
|
r_list_split_iter (list2, iter2);
|
|
|
|
free (iter2);
|
|
|
|
if (r_list_empty (list2)) {
|
|
|
|
r_list_free (list2);
|
|
|
|
r_hashtable64_remove (f->ht_off, item2->offset);
|
|
|
|
r_hashtable64_insert (f->ht_off, item2->offset, NULL);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
lol = r_hashtable64_lookup (f->ht_off, off);
|
|
|
|
if (lol == NULL) {
|
|
|
|
lol = r_list_new ();
|
|
|
|
r_hashtable64_insert (f->ht_off, off, lol);
|
|
|
|
}
|
|
|
|
r_list_append (lol, item);
|
|
|
|
/* update new entry */
|
|
|
|
item->offset = off;
|
|
|
|
item->size = size;
|
|
|
|
} else {
|
|
|
|
item = R_NEW0 (RFlagItem);
|
2013-04-16 01:48:03 +02:00
|
|
|
if (!r_flag_item_set_name (item, name)) {
|
|
|
|
eprintf ("Invalid flag name '%s'.\n", name);
|
|
|
|
free (item);
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
2012-11-14 03:25:32 +01:00
|
|
|
item->space = f->space_idx;
|
|
|
|
r_list_append (f->flags, item);
|
|
|
|
item->offset = off + f->base;
|
|
|
|
item->size = size;
|
|
|
|
|
|
|
|
list = r_hashtable64_lookup (f->ht_name, item->namehash);
|
|
|
|
if (!list) {
|
|
|
|
list = r_list_new ();
|
|
|
|
r_hashtable64_insert (f->ht_name, item->namehash, list);
|
|
|
|
}
|
|
|
|
r_list_append (list, item);
|
|
|
|
|
|
|
|
list2 = r_hashtable64_lookup (f->ht_off, off);
|
|
|
|
if (list2 == NULL) {
|
|
|
|
list2 = r_list_new ();
|
|
|
|
r_hashtable64_insert (f->ht_off, off, list2);
|
|
|
|
}
|
|
|
|
r_list_append (list2, item);
|
2010-06-21 01:58:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-11-14 03:25:32 +01:00
|
|
|
R_API void r_flag_item_set_comment(RFlagItem *item, const char *comment) {
|
|
|
|
free (item->comment);
|
|
|
|
item->comment = strdup (comment);
|
|
|
|
}
|
2010-05-24 11:15:32 +02:00
|
|
|
|
2012-11-14 03:25:32 +01:00
|
|
|
R_API int r_flag_item_set_name(RFlagItem *item, const char *name) {
|
|
|
|
int len;
|
|
|
|
if (!item || !r_name_check (name))
|
|
|
|
return R_FALSE;
|
|
|
|
strncpy (item->name, name, R_FLAG_NAME_SIZE);
|
|
|
|
len = R_MIN (R_FLAG_NAME_SIZE, strlen (r_str_chop (item->name)) + 1);
|
|
|
|
memmove (item->name, r_str_chop (item->name), len);
|
|
|
|
r_name_filter (item->name, 0);
|
|
|
|
item->name[R_FLAG_NAME_SIZE-1]='\0';
|
|
|
|
item->namehash = r_str_hash64 (item->name);
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_flag_rename(RFlag *f, RFlagItem *item, const char *name) {
|
|
|
|
ut64 hash;
|
|
|
|
RList *list;
|
|
|
|
if (!f || !item || !name || !*name) {
|
|
|
|
eprintf ("r_flag_rename: contract fail\n");
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
hash = r_str_hash64 (item->name);
|
|
|
|
list = r_hashtable64_lookup (f->ht_name, hash);
|
|
|
|
if (list) {
|
|
|
|
RFlagItem *item = r_list_get_top (list);
|
|
|
|
if (r_list_empty (list)) {
|
2013-12-03 00:14:59 +01:00
|
|
|
//r_list_free (list);
|
2012-11-14 03:25:32 +01:00
|
|
|
r_hashtable64_remove (f->ht_name, hash);
|
2013-12-03 00:14:59 +01:00
|
|
|
} else {
|
|
|
|
r_hashtable64_remove (f->ht_name, hash);
|
|
|
|
r_list_delete_data (list, item);
|
2010-05-24 11:15:32 +02:00
|
|
|
}
|
2012-11-14 03:25:32 +01:00
|
|
|
if (!r_flag_item_set_name (item, name)) {
|
|
|
|
r_list_append (list, item);
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
list = r_hashtable64_lookup (f->ht_name, item->namehash);
|
|
|
|
if (!list) {
|
|
|
|
list = r_list_new ();
|
|
|
|
r_hashtable64_insert (f->ht_name, item->namehash, list);
|
|
|
|
}
|
|
|
|
r_list_append (list, item);
|
2009-04-04 19:38:59 +00:00
|
|
|
}
|
2012-11-14 03:25:32 +01:00
|
|
|
return R_TRUE;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
|
2012-11-14 03:25:32 +01:00
|
|
|
R_API int r_flag_unset_i(RFlag *f, ut64 off, RFlagItem *p) {
|
|
|
|
RFlagItem *flag = r_flag_get_i (f, off);
|
|
|
|
if (flag) {
|
|
|
|
r_flag_unset (f, flag->name, NULL); //, flag);
|
|
|
|
return R_TRUE;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2012-11-14 03:25:32 +01:00
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_flag_unset_glob(RFlag *f, const char *glob) {
|
|
|
|
int n = 0;
|
|
|
|
RListIter it, *iter;
|
|
|
|
RFlagItem *flag;
|
|
|
|
r_list_foreach (f->flags, iter, flag) {
|
|
|
|
if ((f->space_idx != -1) && (flag->space != f->space_idx))
|
|
|
|
continue;
|
|
|
|
if (r_str_glob (flag->name, glob)) {
|
|
|
|
it.n = iter->n;
|
|
|
|
r_flag_unset (f, flag->name, flag);
|
|
|
|
iter = ⁢
|
|
|
|
n++;
|
2009-06-15 02:44:05 +00:00
|
|
|
}
|
|
|
|
}
|
2012-11-14 03:25:32 +01:00
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void unflag(RFlag *f, ut64 namehash) {
|
|
|
|
RFlagItem *item;
|
|
|
|
RListIter *iter;
|
|
|
|
/* No _safe loop necessary because we return immediately after the delete. */
|
|
|
|
r_list_foreach (f->flags, iter, item) {
|
|
|
|
if (item->namehash == namehash) {
|
|
|
|
r_list_delete (f->flags, iter);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_flag_unset(RFlag *f, const char *name, RFlagItem *p) {
|
|
|
|
ut64 off;
|
|
|
|
RListIter *iter2;
|
|
|
|
RFlagItem *item2, *item = p;
|
2010-06-25 11:22:14 +02:00
|
|
|
ut64 hash = r_str_hash64 (name);
|
2012-11-14 03:25:32 +01:00
|
|
|
RList *list2, *list = r_hashtable64_lookup (f->ht_name, hash);
|
|
|
|
// list = name hash
|
|
|
|
// list2 = off hash
|
|
|
|
if (list && list->head) {
|
|
|
|
if (!item) item = r_list_pop (list);
|
|
|
|
if (!item) return R_FALSE;
|
|
|
|
off = item->offset;
|
2011-05-21 21:05:21 +02:00
|
|
|
|
2012-11-14 03:25:32 +01:00
|
|
|
list2 = r_hashtable64_lookup (f->ht_off, off);
|
|
|
|
if (list2) {
|
|
|
|
/* delete flag by name */
|
|
|
|
/* No _safe loop necessary because we break immediately after the delete. */
|
|
|
|
r_list_foreach (list2, iter2, item2) {
|
|
|
|
if (hash == item2->namehash) {
|
|
|
|
r_list_delete (list2, iter2);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (list2 && r_list_empty (list2)) {
|
|
|
|
r_list_free (list2);
|
|
|
|
r_hashtable64_remove (f->ht_off, off);
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
}
|
2012-11-14 03:25:32 +01:00
|
|
|
/* delete from f->flags list */
|
|
|
|
unflag (f, hash);
|
|
|
|
if (list && r_list_empty (list)) {
|
|
|
|
r_list_free (list);
|
|
|
|
r_hashtable64_remove (f->ht_name, hash);
|
|
|
|
}
|
|
|
|
return R_TRUE;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2012-11-14 03:25:32 +01:00
|
|
|
return R_FALSE;
|
|
|
|
}
|
2009-02-05 22:08:46 +01:00
|
|
|
|
2012-11-14 03:25:32 +01:00
|
|
|
R_API RFlagItem *r_flag_get_at(RFlag *f, ut64 off) {
|
|
|
|
RFlagItem *item, *nice = NULL;
|
|
|
|
RListIter *iter;
|
2009-02-05 22:08:46 +01:00
|
|
|
|
2012-11-14 03:25:32 +01:00
|
|
|
r_list_foreach (f->flags, iter, item) {
|
|
|
|
if (item->offset == off)
|
|
|
|
return item;
|
|
|
|
if (off > item->offset) {
|
|
|
|
if (nice) {
|
|
|
|
if (nice->offset < item->offset)
|
|
|
|
nice = item;
|
|
|
|
} else nice = item;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nice;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2010-07-20 23:25:15 +02:00
|
|
|
|
2013-08-29 03:19:16 +02:00
|
|
|
R_API int r_flag_relocate (RFlag *f, ut64 off, ut64 off_mask, ut64 to) {
|
|
|
|
ut64 neg_mask = ~(off_mask);
|
|
|
|
RFlagItem *item;
|
|
|
|
RListIter *iter;
|
|
|
|
int n = 0;
|
|
|
|
|
|
|
|
r_list_foreach (f->flags, iter, item) {
|
|
|
|
ut64 fn = item->offset & neg_mask;
|
|
|
|
ut64 on = off & neg_mask;
|
|
|
|
if (fn == on) {
|
|
|
|
ut64 fm = item->offset & off_mask;
|
|
|
|
ut64 om = to & off_mask;
|
|
|
|
item->offset = (to&neg_mask) + fm + om;
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
2012-11-14 03:25:32 +01:00
|
|
|
#ifdef MYTEST
|
|
|
|
int main () {
|
|
|
|
RFlagItem *i;
|
|
|
|
RFlag *f = r_flag_new ();
|
|
|
|
r_flag_set (f, "rip", 0xfff333999000LL, 1, 0);
|
|
|
|
r_flag_set (f, "rip", 0xfff333999002LL, 1, 0);
|
|
|
|
r_flag_unset (f, "rip", NULL);
|
|
|
|
r_flag_set (f, "rip", 3, 4, 0);
|
|
|
|
r_flag_set (f, "rip", 4, 4, 0);
|
|
|
|
r_flag_set (f, "corwp", 300, 4, 0);
|
|
|
|
r_flag_set (f, "barp", 300, 4, 0);
|
|
|
|
r_flag_set (f, "rip", 3, 4, 0);
|
|
|
|
r_flag_set (f, "rip", 4, 4, 0);
|
|
|
|
|
|
|
|
i = r_flag_get (f, "rip");
|
|
|
|
if (i) printf ("nRIP: %p %llx\n", i, i->offset);
|
|
|
|
else printf ("nRIP: null\n");
|
|
|
|
|
|
|
|
i = r_flag_get_i (f, 0xfff333999000LL);
|
|
|
|
if (i) printf ("iRIP: %p %llx\n", i, i->offset);
|
|
|
|
else printf ("iRIP: null\n");
|
2010-07-20 23:25:15 +02:00
|
|
|
}
|
2012-11-13 03:25:07 +01:00
|
|
|
#endif
|