radare2/libr/io/undo.c
pancake f7f82ecbc0 * Implement support for handling av/repag and Function keys in r_cons
* Remove -u flag in r2 (was not implemented, but not needed)
* Added ?v command
* Check if file exists in rarun2
* Fix cmd.vprompt
* Some fixes in io.undo
  - Two test cases to check undo and redo ops
* Added 2 bugs in testsuite
2011-06-05 20:36:22 +02:00

188 lines
4.6 KiB
C

/* radare - LGPL - Copyright 2007-2011 pancake<nopcode.org> */
#include <r_io.h>
#if 0
* TODO:
* - make path of indirections shortr (io->undo.foo is slow) */
* - Plugin changes in write and seeks
* - Per-fd history log
#endif
R_API int r_io_undo_init(struct r_io_t *io) {
io->undo.w_init = 0;
io->undo.w_enable = 0;
io->undo.idx = 0;
io->undo.limit = 0;
io->undo.s_enable = 0;
io->undo.w_enable = 0;
INIT_LIST_HEAD(&(io->undo.w_list));
return R_TRUE;
}
R_API void r_io_undo_enable(struct r_io_t *io, int s, int w) {
io->undo.s_enable = s;
io->undo.w_enable = w;
}
R_API ut64 r_io_sundo_last(struct r_io_t *io) {
return (io->undo.idx>0)?
io->undo.seek[io->undo.idx-2] : io->off;
}
R_API int r_io_sundo(struct r_io_t *io) {
if (io->undo.idx == io->undo.limit)
r_io_sundo_push (io);
io->undo.idx--;
if (io->undo.idx<0)
return io->undo.idx = 0;
io->off = io->undo.seek[io->undo.idx-1];
return R_TRUE;
}
R_API int r_io_sundo_redo(struct r_io_t *io) {
if (io->undo.idx<io->undo.limit) {
io->undo.idx += 1;
if (io->undo.idx>=R_IO_UNDOS) {
io->undo.idx -= 1;
return R_FALSE;
} else io->off = io->undo.seek[io->undo.idx-1];
r_io_sundo(io);
return R_TRUE;
}
return R_FALSE;
}
R_API void r_io_sundo_push(RIO *io) {
ut64 off = (io->va && !r_list_empty (io->sections))?
r_io_section_offset_to_vaddr (io, io->off) : io->off;
if (!io->undo.s_enable)
return;
#if 0
if (io->undo.seek[io->undo.idx-1] == off)
return;
#endif
io->undo.seek[io->undo.idx] = off;
io->undo.idx++;
if (io->undo.idx==R_IO_UNDOS-1) {
io->undo.idx--;
eprintf ("undo limit reached\n");
}
//if (io->undo.limit<io->undo.idx)
io->undo.limit = io->undo.idx;
}
R_API void r_io_sundo_reset(struct r_io_t *io) {
io->undo.idx = 0;
}
R_API void r_io_sundo_list(struct r_io_t *io) {
int i;
if (io->undo.idx>0) {
io->printf ("f undo_idx @ %d\n", io->undo.idx);
for (i=io->undo.idx;i!=0;i--)
io->printf ("f undo_%d @ 0x%"PFMT64x"\n",
io->undo.idx-i, io->undo.seek[i-1]);
} else eprintf("-no seeks done-\n");
}
/* undo writez */
R_API void r_io_wundo_new(struct r_io_t *io, ut64 off, const ut8 *data, int len) {
struct r_io_undo_w_t *uw;
if (!io->undo.w_enable)
return;
/* undo write changes */
uw = R_NEW(struct r_io_undo_w_t);
if (!uw) return;
uw->set = R_TRUE;
uw->off = off;
uw->len = len;
uw->n = (ut8*) malloc(len);
memcpy(uw->n, data, len);
uw->o = (ut8*) malloc(len);
r_io_read_at(io, off, uw->o, len);
list_add_tail(&(uw->list), &(io->undo.w_list));
}
R_API void r_io_wundo_clear(struct r_io_t *io) {
// XXX memory leak
INIT_LIST_HEAD(&(io->undo.w_list));
}
// rename to r_io_undo_length ?
R_API int r_io_wundo_size(struct r_io_t *io) {
struct list_head *p;
int i = 0;
if (io->undo.w_init)
list_for_each_prev(p, &(io->undo.w_list))
i++;
return i;
}
// TODO: Deprecate or so? iterators must be language-wide, but helpers are useful
R_API void r_io_wundo_list(struct r_io_t *io) {
#define BW 8 /* byte wrap */
struct list_head *p;
int i = 0, j, len;
if (io->undo.w_init)
list_for_each_prev(p, &(io->undo.w_list)) {
struct r_io_undo_w_t *u = list_entry(p, struct r_io_undo_w_t, list);
io->printf ("%02d %c %d %08"PFMT64x": ", i, u->set?'+':'-', u->len, u->off);
len = (u->len>BW)?BW:u->len;
for(j=0;j<len;j++) io->printf ("%02x ", u->o[j]);
if (len == BW) io->printf (".. ");
io->printf ("=> ");
for(j=0;j<len;j++) io->printf ("%02x ", u->n[j]);
if (len == BW) io->printf (".. ");
io->printf ("\n");
i++;
}
}
R_API int r_io_wundo_apply(struct r_io_t *io, struct r_io_undo_w_t *u, int set) {
int orig = io->undo.w_enable;
io->undo.w_enable = 0;
if (set) {
r_io_write_at(io, u->off, u->n, u->len);
u->set = R_TRUE;
} else {
r_io_write_at(io, u->off, u->o, u->len);
u->set = R_FALSE;
}
io->undo.w_enable = orig;
return 0;
}
R_API void r_io_wundo_apply_all(struct r_io_t *io, int set) {
struct list_head *p;
list_for_each_prev (p, &(io->undo.w_list)) {
struct r_io_undo_w_t *u = list_entry (p, struct r_io_undo_w_t, list);
r_io_wundo_apply (io, u, set); //UNDO_WRITE_UNSET);
eprintf ("%s 0x%08"PFMT64x"\n", set?"redo":"undo", u->off);
}
}
/* sets or unsets the writes done */
/* if ( set == 0 ) unset(n) */
R_API int r_io_wundo_set(struct r_io_t *io, int n, int set) {
struct r_io_undo_w_t *u = NULL;
struct list_head *p;
int i = 0;
if (io->undo.w_init) {
list_for_each_prev(p, &(io->undo.w_list)) {
if (i++ == n) {
u = list_entry(p, struct r_io_undo_w_t, list);
break;
}
}
if (u) {
r_io_wundo_apply(io, u, set);
return R_TRUE;
} else eprintf ("invalid undo-write index\n");
} else eprintf ("no writes done\n");
return R_FALSE;
}