mirror of
https://github.com/radareorg/radare2.git
synced 2024-11-27 23:20:40 +00:00
3b85115dd3
* Use strcmp or r_str_startswith instead
1387 lines
34 KiB
C
1387 lines
34 KiB
C
/* radare - LGPL - Copyright 2019-2023 - pancake */
|
|
|
|
#include <r_util/r_table.h>
|
|
#include "r_cons.h"
|
|
|
|
// cant do that without globals because RList doesnt have void *user :(
|
|
static R_TH_LOCAL int Gnth = 0;
|
|
static R_TH_LOCAL RListComparator Gcmp = NULL;
|
|
|
|
static int sortString(const void *a, const void *b) {
|
|
return strcmp (a, b);
|
|
}
|
|
|
|
static int sortNumber(const void *a, const void *b) {
|
|
return r_num_get (NULL, a) - r_num_get (NULL, b);
|
|
}
|
|
|
|
// maybe just index by name instead of exposing those symbols as global
|
|
static RTableColumnType r_table_type_string = { "string", sortString };
|
|
static RTableColumnType r_table_type_number = { "number", sortNumber };
|
|
static RTableColumnType r_table_type_bool = { "bool", sortNumber };
|
|
|
|
R_API RTableColumnType *r_table_type(const char *name) {
|
|
if (r_str_startswith (name, "bool")) {
|
|
return &r_table_type_bool;
|
|
}
|
|
if (!strcmp (name, "string")) {
|
|
return &r_table_type_string;
|
|
}
|
|
if (!strcmp (name, "number")) {
|
|
return &r_table_type_number;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
// TODO: unused for now, maybe good to call after filter :?
|
|
static void __table_adjust(RTable *t) {
|
|
RListIter *iter, *iter2;
|
|
RTableColumn *col;
|
|
RTableRow *row;
|
|
r_list_foreach (t->cols, iter, col) {
|
|
int itemLength = r_str_len_utf8_ansi (col->name) + 1;
|
|
col->width = itemLength;
|
|
}
|
|
r_list_foreach (t->rows, iter, row) {
|
|
const char *item;
|
|
int ncol = 0;
|
|
r_list_foreach (row->items, iter2, item) {
|
|
int itemLength = r_str_len_utf8_ansi (item) + 1;
|
|
RTableColumn *c = r_list_get_n (t->cols, ncol);
|
|
if (c) {
|
|
c->width = R_MAX (c->width, itemLength);
|
|
}
|
|
ncol ++;
|
|
}
|
|
}
|
|
}
|
|
|
|
R_API void r_table_row_free(void *_row) {
|
|
RTableRow *row = _row;
|
|
r_list_free (row->items);
|
|
free (row);
|
|
}
|
|
|
|
R_API void r_table_column_free(void *_col) {
|
|
RTableColumn *col = _col;
|
|
free (col->name);
|
|
free (col);
|
|
}
|
|
|
|
R_API RTableRow *r_table_row_clone(RTableRow *row) {
|
|
RTableRow *r = r_table_row_new (r_list_newf (free));
|
|
RListIter *iter;
|
|
char *word;
|
|
r_list_foreach (row->items, iter, word) {
|
|
r_list_append (r->items, strdup (word));
|
|
}
|
|
return r;
|
|
}
|
|
|
|
R_API RTableColumn *r_table_column_clone(RTableColumn *col) {
|
|
RTableColumn *c = R_NEW0 (RTableColumn);
|
|
if (!c) {
|
|
return NULL;
|
|
}
|
|
memcpy (c, col, sizeof (*c));
|
|
c->name = strdup (c->name);
|
|
return c;
|
|
}
|
|
|
|
R_API RTable *r_table_new(const char *name) {
|
|
RTable *t = R_NEW0 (RTable);
|
|
if (t) {
|
|
t->showHeader = true;
|
|
t->name = strdup (name);
|
|
t->cols = r_list_newf (r_table_column_free);
|
|
t->rows = r_list_newf (r_table_row_free);
|
|
t->showSum = false;
|
|
}
|
|
return t;
|
|
}
|
|
|
|
R_API void r_table_free(RTable *t) {
|
|
if (!t) {
|
|
return;
|
|
}
|
|
r_list_free (t->cols);
|
|
r_list_free (t->rows);
|
|
free (t->name);
|
|
free (t);
|
|
}
|
|
|
|
R_API void r_table_add_column(RTable *t, RTableColumnType *type, const char *name, int maxWidth) {
|
|
RTableColumn *c = R_NEW0 (RTableColumn);
|
|
if (c) {
|
|
c->name = strdup (name);
|
|
c->maxWidth = maxWidth;
|
|
c->type = type;
|
|
int itemLength = r_str_len_utf8_ansi (name) + 1;
|
|
c->width = itemLength;
|
|
r_list_append (t->cols, c);
|
|
c->total = -1;
|
|
}
|
|
}
|
|
|
|
R_API RTableRow *r_table_row_new(RList *items) {
|
|
RTableRow *row = R_NEW (RTableRow);
|
|
row->items = items;
|
|
return row;
|
|
}
|
|
|
|
static bool __addRow(RTable *t, RList *items, const char *arg, int col) {
|
|
int itemLength = r_str_len_utf8_ansi (arg) + 1;
|
|
RTableColumn *c = r_list_get_n (t->cols, col);
|
|
if (c) {
|
|
c->width = R_MAX (c->width, itemLength);
|
|
r_list_append (items, strdup (arg));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
R_API void r_table_add_row_list(RTable *t, RList *items) {
|
|
r_return_if_fail (t && items);
|
|
RTableRow *row = r_table_row_new (items);
|
|
r_list_append (t->rows, row);
|
|
// throw warning if not enough columns defined in header
|
|
t->totalCols = R_MAX (t->totalCols, r_list_length (items));
|
|
}
|
|
|
|
R_API void r_table_set_columnsf(RTable *t, const char *fmt, ...) {
|
|
va_list ap;
|
|
va_start (ap, fmt);
|
|
RTableColumnType *typeString = r_table_type ("string");
|
|
RTableColumnType *typeNumber = r_table_type ("number");
|
|
RTableColumnType *typeBool = r_table_type ("bool");
|
|
const char *name;
|
|
const char *f = fmt;
|
|
for (;*f;f++) {
|
|
name = va_arg (ap, const char *);
|
|
if (!name) {
|
|
break;
|
|
}
|
|
switch (*f) {
|
|
case 'b':
|
|
r_table_add_column (t, typeBool, name, 0);
|
|
break;
|
|
case 's':
|
|
case 'z':
|
|
r_table_add_column (t, typeString, name, 0);
|
|
break;
|
|
case 'i':
|
|
case 'd':
|
|
case 'n':
|
|
case 'x':
|
|
case 'X':
|
|
r_table_add_column (t, typeNumber, name, 0);
|
|
break;
|
|
default:
|
|
R_LOG_ERROR ("Invalid format string char '%c', use 's' or 'n'", *f);
|
|
break;
|
|
}
|
|
}
|
|
va_end (ap);
|
|
}
|
|
|
|
R_API void r_table_add_rowf(RTable *t, const char *fmt, ...) {
|
|
va_list ap;
|
|
va_start (ap, fmt);
|
|
RList *list = r_list_newf (free);
|
|
const char *f = fmt;
|
|
const char *arg = NULL;
|
|
for (; *f; f++) {
|
|
switch (*f) {
|
|
case 's':
|
|
case 'z':
|
|
arg = va_arg (ap, const char *);
|
|
r_list_append (list, strdup (r_str_get (arg)));
|
|
break;
|
|
case 'b':
|
|
r_list_append (list, r_str_new (r_str_bool (va_arg (ap, int))));
|
|
break;
|
|
case 'i':
|
|
case 'd':
|
|
r_list_append (list, r_str_newf ("%d", va_arg (ap, int)));
|
|
break;
|
|
case 'n':
|
|
r_list_append (list, r_str_newf ("%"PFMT64d, va_arg (ap, ut64)));
|
|
break;
|
|
case 'u':
|
|
r_list_append (list, r_num_units (NULL, 32, va_arg (ap, ut64)));
|
|
break;
|
|
case 'x':
|
|
case 'X':
|
|
{
|
|
ut64 n = va_arg (ap, ut64);
|
|
if (n == UT64_MAX) {
|
|
if (*f == 'X') {
|
|
r_list_append (list, strdup ("----------"));
|
|
} else {
|
|
r_list_append (list, strdup ("-1"));
|
|
}
|
|
} else {
|
|
if (*f == 'X') {
|
|
r_list_append (list, r_str_newf ("0x%08"PFMT64x, n));
|
|
} else {
|
|
r_list_append (list, r_str_newf ("0x%"PFMT64x, n));
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
R_LOG_ERROR ("Invalid format string char '%c', use 's' or 'n'", *f);
|
|
break;
|
|
}
|
|
}
|
|
va_end (ap);
|
|
r_table_add_row_list (t, list);
|
|
}
|
|
|
|
R_API void r_table_add_row(RTable *t, const char *name, ...) {
|
|
va_list ap;
|
|
va_start (ap, name);
|
|
int col = 0;
|
|
RList *items = r_list_newf (free);
|
|
__addRow (t, items, name, col++);
|
|
for (;;) {
|
|
const char *arg = va_arg (ap, const char *);
|
|
if (!arg) {
|
|
break;
|
|
}
|
|
__addRow (t, items, arg, col);
|
|
// TODO: assert if number of columns doesnt match t->cols
|
|
col++;
|
|
}
|
|
va_end (ap);
|
|
RTableRow *row = r_table_row_new (items);
|
|
r_list_append (t->rows, row);
|
|
// throw warning if not enough columns defined in header
|
|
t->totalCols = R_MAX (t->totalCols, r_list_length (items));
|
|
}
|
|
|
|
// import / export
|
|
|
|
static int __strbuf_append_col_aligned_fancy(RTable *t, RStrBuf *sb, RTableColumn *col, char *str) {
|
|
RCons *cons = (RCons *) t->cons;
|
|
const char *v_line = (cons && (cons->use_utf8 || cons->use_utf8_curvy)) ? RUNE_LINE_VERT : "|";
|
|
int ll = r_strbuf_length (sb);
|
|
switch (col->align) {
|
|
case R_TABLE_ALIGN_LEFT:
|
|
r_strbuf_appendf (sb, "%s %-*s ", v_line, col->width, str);
|
|
break;
|
|
case R_TABLE_ALIGN_RIGHT:
|
|
r_strbuf_appendf (sb, "%s %*s ", v_line, col->width, str);
|
|
break;
|
|
case R_TABLE_ALIGN_CENTER:
|
|
{
|
|
int len = r_str_len_utf8 (str);
|
|
int pad = (col->width - len) / 2;
|
|
int left = col->width - (pad * 2 + len);
|
|
r_strbuf_appendf (sb, "%s %-*s ", v_line, pad, " ");
|
|
r_strbuf_appendf (sb, "%-*s ", pad + left, str);
|
|
}
|
|
break;
|
|
}
|
|
return r_strbuf_length (sb) - ll;
|
|
}
|
|
|
|
static void __computeTotal(RTable *t) {
|
|
RTableRow *row;
|
|
RListIter *iter, *iter2;
|
|
r_list_foreach (t->rows, iter, row) {
|
|
char *item;
|
|
int c = 0;
|
|
r_list_foreach (row->items, iter2, item) {
|
|
RTableColumn *col = r_list_get_n (t->cols, c);
|
|
if (r_str_startswith (col->type->name, "number") && r_str_isnumber (item)) {
|
|
if (col->total < 0) {
|
|
col->total = 0;
|
|
}
|
|
col->total += sdb_atoi(item);
|
|
}
|
|
c++;
|
|
}
|
|
}
|
|
}
|
|
|
|
R_API char *r_table_tofancystring(RTable *t) {
|
|
if (r_list_length (t->cols) == 0) {
|
|
return strdup ("");
|
|
}
|
|
RStrBuf *sb = r_strbuf_new ("");
|
|
RTableRow *row;
|
|
RTableColumn *col;
|
|
RCons *cons = (RCons *)t->cons;
|
|
RListIter *iter, *iter2;
|
|
bool useUtf8 = (cons && cons->use_utf8);
|
|
bool useUtf8Curvy = (cons && cons->use_utf8_curvy);
|
|
const char *v_line = useUtf8 || useUtf8Curvy ? RUNE_LINE_VERT : "|";
|
|
const char *h_line = useUtf8 || useUtf8Curvy ? RUNE_LINE_HORIZ : "-";
|
|
const char *l_intersect = useUtf8 || useUtf8Curvy ? RUNE_LINE_VERT : ")";
|
|
const char *r_intersect = useUtf8 || useUtf8Curvy ? RUNE_LINE_VERT : "(";
|
|
const char *tl_corner = useUtf8 ? (useUtf8Curvy ? RUNE_CURVE_CORNER_TL : RUNE_CORNER_TL) : ".";
|
|
const char *tr_corner = useUtf8 ? (useUtf8Curvy ? RUNE_CURVE_CORNER_TR : RUNE_CORNER_TR) : ".";
|
|
const char *bl_corner = useUtf8 ? (useUtf8Curvy ? RUNE_CURVE_CORNER_BL : RUNE_CORNER_BL) : "`";
|
|
const char *br_corner = useUtf8 ? (useUtf8Curvy ? RUNE_CURVE_CORNER_BR : RUNE_CORNER_BR) : "'";
|
|
__table_adjust (t);
|
|
|
|
r_list_foreach (t->cols, iter, col) {
|
|
__strbuf_append_col_aligned_fancy (t, sb, col, col->name);
|
|
}
|
|
int len = r_str_len_utf8_ansi (r_strbuf_get (sb)) - 1;
|
|
int maxlen = len;
|
|
char *h_line_str = r_str_repeat (h_line, maxlen);
|
|
{
|
|
char *s = r_str_newf ("%s%s%s\n", tl_corner, h_line_str, tr_corner);
|
|
r_strbuf_prepend (sb, s);
|
|
free (s);
|
|
}
|
|
|
|
r_strbuf_appendf (sb, "%s\n%s%s%s\n", v_line, l_intersect, h_line_str, r_intersect);
|
|
r_list_foreach (t->rows, iter, row) {
|
|
char *item;
|
|
int c = 0;
|
|
r_list_foreach (row->items, iter2, item) {
|
|
RTableColumn *col = r_list_get_n (t->cols, c);
|
|
if (col) {
|
|
int l = __strbuf_append_col_aligned_fancy (t, sb, col, item);
|
|
len = R_MAX (len, l);
|
|
}
|
|
c++;
|
|
}
|
|
r_strbuf_appendf (sb, "%s\n", v_line);
|
|
}
|
|
|
|
if (t->showSum) {
|
|
char tmp[SDB_NUM_BUFSZ];
|
|
__computeTotal (t);
|
|
r_strbuf_appendf (sb, "%s%s%s\n", l_intersect, h_line_str, r_intersect);
|
|
r_list_foreach (t->cols, iter, col) {
|
|
char *num = col->total == -1 ? "" : sdb_itoa (col->total, 10, tmp, sizeof (tmp));
|
|
int l = __strbuf_append_col_aligned_fancy (t, sb, col, num);
|
|
len = R_MAX (len, l);
|
|
}
|
|
r_strbuf_appendf (sb, "%s\n", v_line);
|
|
}
|
|
r_strbuf_appendf (sb, "%s%s%s\n", bl_corner, h_line_str, br_corner);
|
|
free (h_line_str);
|
|
return r_strbuf_drain (sb);
|
|
}
|
|
|
|
static int __strbuf_append_col_aligned(RStrBuf *sb, RTableColumn *col, const char *str, bool nopad) {
|
|
int ll = r_strbuf_length (sb);
|
|
if (nopad) {
|
|
r_strbuf_appendf (sb, "%s", str);
|
|
} else {
|
|
char *pad = "";
|
|
int len = r_str_len_utf8_ansi (str);
|
|
int padlen = (len < col->width)? col->width - len: 0;
|
|
switch (col->align) {
|
|
case R_TABLE_ALIGN_LEFT:
|
|
pad = r_str_repeat (" ", col->width - len);
|
|
r_strbuf_append (sb, str);
|
|
r_strbuf_append (sb, pad);
|
|
free (pad);
|
|
break;
|
|
case R_TABLE_ALIGN_RIGHT:
|
|
pad = r_str_repeat (" ", padlen);
|
|
r_strbuf_append (sb, pad);
|
|
r_strbuf_append (sb, str);
|
|
r_strbuf_append (sb, " ");
|
|
free (pad);
|
|
break;
|
|
case R_TABLE_ALIGN_CENTER:
|
|
{
|
|
int pad = (col->width - len) / 2;
|
|
int left = col->width - pad - len;
|
|
r_strbuf_appendf (sb, "%-*s", pad, " ");
|
|
r_strbuf_appendf (sb, "%s ", str);
|
|
r_strbuf_appendf (sb, "%-*s", left, " ");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return r_strbuf_length (sb) - ll;
|
|
}
|
|
|
|
R_API char *r_table_tostring(RTable *t) {
|
|
if (!t) { // guard
|
|
return strdup ("");
|
|
}
|
|
if (t->showR2) {
|
|
return r_table_tor2cmds (t);
|
|
}
|
|
if (t->showSQL) {
|
|
return r_table_tosql (t);
|
|
}
|
|
if (t->showTSV) {
|
|
return r_table_totsv (t);
|
|
}
|
|
if (t->showCSV) {
|
|
return r_table_tocsv (t);
|
|
}
|
|
if (t->showHTML) {
|
|
return r_table_tohtml (t);
|
|
}
|
|
if (t->showJSON) {
|
|
char *s = r_table_tojson (t);
|
|
char *q = r_str_newf ("%s\n", s);
|
|
free (s);
|
|
return q;
|
|
}
|
|
if (t->showFancy) {
|
|
return r_table_tofancystring (t);
|
|
}
|
|
return r_table_tosimplestring (t);
|
|
}
|
|
|
|
static bool nopad_trailing(RListIter *iter) {
|
|
while (iter->n) {
|
|
iter = iter->n;
|
|
char *next_item = iter->data;
|
|
if (R_STR_ISNOTEMPTY (next_item)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
R_API char *r_table_tosimplestring(RTable *t) {
|
|
RStrBuf *sb = r_strbuf_new ("");
|
|
RTableRow *row;
|
|
RTableColumn *col;
|
|
RListIter *iter, *iter2;
|
|
RCons *cons = (RCons *) t->cons;
|
|
const char *h_line = (cons && (cons->use_utf8 || cons->use_utf8_curvy)) ? RUNE_LONG_LINE_HORIZ : "-";
|
|
__table_adjust (t);
|
|
int maxlen = 0;
|
|
if (t->showHeader) {
|
|
r_list_foreach (t->cols, iter, col) {
|
|
bool nopad = !iter->n;
|
|
int ll = __strbuf_append_col_aligned (sb, col, col->name, nopad);
|
|
maxlen = R_MAX (maxlen, ll);
|
|
}
|
|
int len = r_str_len_utf8_ansi (r_strbuf_get (sb));
|
|
char *l = r_str_repeat (h_line, R_MAX (maxlen, len));
|
|
if (R_LIKELY (l)) {
|
|
r_strbuf_appendf (sb, "\n%s\n", l);
|
|
free (l);
|
|
}
|
|
}
|
|
r_list_foreach (t->rows, iter, row) {
|
|
char *item;
|
|
int c = 0;
|
|
r_list_foreach (row->items, iter2, item) {
|
|
bool nopad = nopad_trailing (iter2);
|
|
RTableColumn *col = r_list_get_n (t->cols, c);
|
|
if (R_LIKELY (col)) {
|
|
(void)__strbuf_append_col_aligned (sb, col, item, nopad);
|
|
}
|
|
c++;
|
|
}
|
|
r_strbuf_append (sb, "\n");
|
|
}
|
|
if (t->showSum) {
|
|
char tmp[SDB_NUM_BUFSZ];
|
|
__computeTotal (t);
|
|
if (maxlen > 0) {
|
|
char *l = r_str_repeat (h_line, maxlen);
|
|
if (l) {
|
|
r_strbuf_appendf (sb, "\n%s\n", l);
|
|
free (l);
|
|
}
|
|
}
|
|
r_list_foreach (t->cols, iter, col) {
|
|
bool nopad = !iter->n;
|
|
(void)__strbuf_append_col_aligned (sb, col, sdb_itoa (col->total, 10, tmp, sizeof (tmp)), nopad);
|
|
}
|
|
}
|
|
return r_strbuf_drain (sb);
|
|
}
|
|
|
|
R_API char *r_table_tor2cmds(RTable *t) {
|
|
RStrBuf *sb = r_strbuf_new ("");
|
|
RTableRow *row;
|
|
RTableColumn *col;
|
|
RListIter *iter, *iter2;
|
|
|
|
r_strbuf_append (sb, ",h ");
|
|
r_list_foreach (t->cols, iter, col) {
|
|
char fmt = col->type == &r_table_type_string? 's': 'x';
|
|
r_strbuf_appendf (sb, "%c", fmt);
|
|
}
|
|
r_list_foreach (t->cols, iter, col) {
|
|
r_strbuf_appendf (sb, " %s", col->name);
|
|
}
|
|
r_strbuf_append (sb, "\n");
|
|
|
|
r_list_foreach (t->rows, iter, row) {
|
|
char *item;
|
|
int c = 0;
|
|
r_strbuf_append (sb, ",r");
|
|
r_list_foreach (row->items, iter2, item) {
|
|
RTableColumn *col = r_list_get_n (t->cols, c);
|
|
if (col) {
|
|
r_strbuf_append (sb, " ");
|
|
r_strbuf_append (sb, item);
|
|
}
|
|
c++;
|
|
}
|
|
r_strbuf_append (sb, "\n");
|
|
}
|
|
return r_strbuf_drain (sb);
|
|
}
|
|
|
|
R_API char *r_table_tosql(RTable *t) {
|
|
r_return_val_if_fail (t, NULL);
|
|
RStrBuf *sb = r_strbuf_new ("");
|
|
RTableRow *row;
|
|
RTableColumn *col;
|
|
RListIter *iter, *iter2;
|
|
|
|
const char *table_name = R_STR_ISEMPTY (t->name)? "r2": t->name;
|
|
r_strbuf_appendf (sb, "CREATE TABLE %s (", table_name);
|
|
bool primary_key = true;
|
|
r_list_foreach (t->cols, iter, col) {
|
|
const char *type = col->type == &r_table_type_string? "VARCHAR": "NUMERIC(20)";
|
|
const char *comma = iter->n? ", ": "";
|
|
const char *pkey = primary_key? " PRIMARY KEY": "";
|
|
char *s = r_str_escape_sql (col->name);
|
|
r_strbuf_appendf (sb, "%s %s%s%s", s, type, pkey, comma);
|
|
free (s);
|
|
primary_key = false;
|
|
}
|
|
r_strbuf_append (sb, ");\n");
|
|
|
|
r_list_foreach (t->rows, iter, row) {
|
|
const char *item;
|
|
int c = 0;
|
|
r_strbuf_appendf (sb, "INSERT INTO %s (", table_name);
|
|
r_list_foreach (t->cols, iter2, col) {
|
|
char *s = r_str_escape_sql (col->name);
|
|
r_strbuf_append (sb, s);
|
|
if (iter2->n) {
|
|
r_strbuf_append (sb, ", ");
|
|
}
|
|
free (s);
|
|
}
|
|
r_strbuf_append (sb, ") VALUES (");
|
|
r_list_foreach (row->items, iter2, item) {
|
|
RTableColumn *col = r_list_get_n (t->cols, c);
|
|
if (col) {
|
|
const char *comma = iter2->n? ", ": "";
|
|
if (col->type == &r_table_type_string) {
|
|
char *s = r_str_escape_sql (item);
|
|
r_strbuf_appendf (sb, "'%s'%s", s, comma);
|
|
free (s);
|
|
} else {
|
|
r_strbuf_appendf (sb, "%s%s", item, comma);
|
|
}
|
|
}
|
|
c++;
|
|
}
|
|
r_strbuf_append (sb, ");\n");
|
|
}
|
|
return r_strbuf_drain (sb);
|
|
}
|
|
|
|
static char *tocsv(RTable *t, const char *sep) {
|
|
RStrBuf *sb = r_strbuf_new ("");
|
|
RTableRow *row;
|
|
RTableColumn *col;
|
|
RListIter *iter, *iter2;
|
|
if (t->showHeader) {
|
|
const char *comma = "";
|
|
r_list_foreach (t->cols, iter, col) {
|
|
if (strchr (col->name, *sep)) {
|
|
// TODO. escaped string?
|
|
r_strbuf_appendf (sb, "%s\"%s\"", comma, col->name);
|
|
} else {
|
|
r_strbuf_appendf (sb, "%s%s", comma, col->name);
|
|
}
|
|
comma = sep;
|
|
}
|
|
r_strbuf_append (sb, "\n");
|
|
}
|
|
r_list_foreach (t->rows, iter, row) {
|
|
char *item;
|
|
int c = 0;
|
|
const char *comma = "";
|
|
r_list_foreach (row->items, iter2, item) {
|
|
RTableColumn *col = r_list_get_n (t->cols, c);
|
|
if (col) {
|
|
if (strchr (col->name, *sep)) {
|
|
r_strbuf_appendf (sb, "%s\"%s\"", comma, col->name);
|
|
} else {
|
|
r_strbuf_appendf (sb, "%s%s", comma, item);
|
|
}
|
|
comma = sep;
|
|
}
|
|
c++;
|
|
}
|
|
r_strbuf_append (sb, "\n");
|
|
}
|
|
return r_strbuf_drain (sb);
|
|
}
|
|
|
|
R_API char *r_table_totsv(RTable *t) {
|
|
return tocsv (t, "\t");
|
|
}
|
|
|
|
R_API char *r_table_tocsv(RTable *t) {
|
|
return tocsv (t, ",");
|
|
}
|
|
|
|
R_API char *r_table_tohtml(RTable *t) {
|
|
PJ *pj = pj_new ();
|
|
RTableRow *row;
|
|
RListIter *iter, *iter2;
|
|
pj_a (pj);
|
|
RStrBuf *sb = r_strbuf_new ("");
|
|
r_strbuf_append (sb, "<table>\n");
|
|
// TODO: add th
|
|
r_list_foreach (t->rows, iter, row) {
|
|
char *item;
|
|
r_strbuf_append (sb, " <tr>\n");
|
|
r_list_foreach (row->items, iter2, item) {
|
|
r_strbuf_appendf (sb, " <td>%s</td>\n", item);
|
|
}
|
|
r_strbuf_append (sb, " </tr>\n");
|
|
}
|
|
r_strbuf_append (sb, "</table>\n");
|
|
return r_strbuf_drain (sb);
|
|
}
|
|
|
|
R_API char *r_table_tojson(RTable *t) {
|
|
PJ *pj = pj_new ();
|
|
RTableRow *row;
|
|
RListIter *iter, *iter2;
|
|
pj_a (pj);
|
|
r_list_foreach (t->rows, iter, row) {
|
|
char *item;
|
|
int c = 0;
|
|
pj_o (pj);
|
|
r_list_foreach (row->items, iter2, item) {
|
|
RTableColumn *col = r_list_get_n (t->cols, c);
|
|
if (col) {
|
|
if (col->type == &r_table_type_number) {
|
|
ut64 n = r_num_get (NULL, item);
|
|
if (n) {
|
|
pj_kn (pj, col->name, n);
|
|
} else if (*item && *item != '0') {
|
|
pj_ks (pj, col->name, item);
|
|
}
|
|
} else {
|
|
if (*item) {
|
|
pj_ks (pj, col->name, item);
|
|
}
|
|
}
|
|
}
|
|
c++;
|
|
}
|
|
pj_end (pj);
|
|
}
|
|
pj_end (pj);
|
|
return pj_drain (pj);
|
|
}
|
|
|
|
R_API void r_table_filter(RTable *t, int nth, int op, const char *un) {
|
|
r_return_if_fail (t && un);
|
|
RTableRow *row;
|
|
RListIter *iter, *iter2;
|
|
ut64 uv = r_num_math (NULL, un);
|
|
ut64 sum = 0;
|
|
int page = 0, page_items = 0;
|
|
size_t lrow = 0;
|
|
if (op == 't') {
|
|
size_t ll = r_list_length (t->rows);
|
|
if (ll > uv) {
|
|
uv = ll - uv;
|
|
}
|
|
}
|
|
if (op == 'p') {
|
|
sscanf (un, "%d/%d", &page, &page_items);
|
|
if (page < 1) {
|
|
page = 1;
|
|
}
|
|
if (!ST32_MUL_OVFCHK (page, page_items)) {
|
|
lrow = page_items * (page - 1);
|
|
uv = ((ut64)page_items) * page;
|
|
}
|
|
}
|
|
size_t nrow = 0;
|
|
r_list_foreach_safe (t->rows, iter, iter2, row) {
|
|
const char *nn = r_list_get_n (row->items, nth);
|
|
ut64 nv = r_num_math (NULL, nn);
|
|
bool match = true;
|
|
switch (op) {
|
|
case 'p':
|
|
nrow++;
|
|
if (nrow < lrow) {
|
|
match = false;
|
|
}
|
|
if (nrow > uv) {
|
|
match = false;
|
|
}
|
|
break;
|
|
case 't':
|
|
nrow++;
|
|
if (nrow < uv) {
|
|
match = false;
|
|
}
|
|
break;
|
|
case 'S':
|
|
nrow++;
|
|
match = (nrow > uv);
|
|
break;
|
|
case 'h':
|
|
nrow++;
|
|
if (nrow > uv) {
|
|
match = false;
|
|
}
|
|
break;
|
|
case '+':
|
|
// "sum"
|
|
sum += nv;
|
|
match = false;
|
|
break;
|
|
case '>':
|
|
match = (nv > uv);
|
|
break;
|
|
case '<':
|
|
match = (nv < uv);
|
|
break;
|
|
case '=':
|
|
if (nv == 0) {
|
|
match = (nn && un && !strcmp (nn, un));
|
|
} else {
|
|
match = (nv == uv);
|
|
}
|
|
break;
|
|
case '!':
|
|
if (nv == 0) {
|
|
match = (!nn || !un || strcmp (nn, un));
|
|
} else {
|
|
match = (nv != uv);
|
|
}
|
|
break;
|
|
case '$':
|
|
match = !nn || !un || strstr (nn, un) == NULL;
|
|
break;
|
|
case '~':
|
|
match = nn&&un&&strstr (nn, un);
|
|
break;
|
|
case 's':
|
|
match = strlen (nn) == atoi (un);
|
|
break;
|
|
case 'l':
|
|
match = strlen (nn) > atoi (un);
|
|
break;
|
|
case 'L':
|
|
match = strlen (nn) < atoi (un);
|
|
break;
|
|
case '\0':
|
|
break;
|
|
}
|
|
if (!match) {
|
|
r_list_delete (t->rows, iter);
|
|
}
|
|
}
|
|
if (op == '+') {
|
|
r_table_add_rowf (t, "u", sum);
|
|
}
|
|
}
|
|
|
|
static int cmp(const void *_a, const void *_b) {
|
|
RTableRow *a = (RTableRow*)_a;
|
|
RTableRow *b = (RTableRow*)_b;
|
|
const char *wa = r_list_get_n (a->items, Gnth);
|
|
const char *wb = r_list_get_n (b->items, Gnth);
|
|
int res = Gcmp (wa, wb);
|
|
return res;
|
|
}
|
|
|
|
R_API void r_table_sort(RTable *t, int nth, bool dec) {
|
|
RTableColumn *col = r_list_get_n (t->cols, nth);
|
|
if (col) {
|
|
Gnth = nth;
|
|
if (col->type && col->type->cmp) {
|
|
Gcmp = col->type->cmp;
|
|
t->rows->sorted = false; //force sorting
|
|
r_list_sort (t->rows, cmp);
|
|
if (dec) {
|
|
r_list_reverse (t->rows);
|
|
}
|
|
}
|
|
Gnth = 0;
|
|
Gcmp = NULL;
|
|
}
|
|
}
|
|
|
|
static int cmplen(const void *_a, const void *_b) {
|
|
RTableRow *a = (RTableRow*)_a;
|
|
RTableRow *b = (RTableRow*)_b;
|
|
const char *wa = r_list_get_n (a->items, Gnth);
|
|
const char *wb = r_list_get_n (b->items, Gnth);
|
|
int res = strlen (wa) - strlen (wb);
|
|
return res;
|
|
}
|
|
|
|
R_API void r_table_sortlen(RTable *t, int nth, bool dec) {
|
|
RTableColumn *col = r_list_get_n (t->cols, nth);
|
|
if (col) {
|
|
Gnth = nth;
|
|
t->rows->sorted = false; //force sorting
|
|
r_list_sort (t->rows, cmplen);
|
|
if (dec) {
|
|
r_list_reverse (t->rows);
|
|
}
|
|
Gnth = 0;
|
|
}
|
|
}
|
|
|
|
static int r_rows_cmp(RList *lhs, RList *rhs, RList *cols, int nth) {
|
|
RListIter *iter_lhs;
|
|
RListIter *iter_rhs;
|
|
RListIter *iter_col;
|
|
RTableColumn *item_col;
|
|
|
|
void *item_lhs;
|
|
void *item_rhs;
|
|
int tmp;
|
|
int i = 0;
|
|
|
|
for (iter_lhs = lhs->head, iter_rhs = rhs->head, iter_col = cols->head;
|
|
iter_lhs && iter_rhs && iter_col;
|
|
iter_lhs = iter_lhs->n, iter_rhs = iter_rhs->n, iter_col = iter_col->n) {
|
|
|
|
item_lhs = iter_lhs->data;
|
|
item_rhs = iter_rhs->data;
|
|
item_col = iter_col->data;
|
|
|
|
if (nth == -1 || i == nth) {
|
|
tmp = item_col->type->cmp (item_lhs, item_rhs);
|
|
|
|
if (tmp) {
|
|
return tmp;
|
|
}
|
|
}
|
|
|
|
i++;
|
|
}
|
|
|
|
if (iter_lhs) {
|
|
return 1;
|
|
}
|
|
|
|
if (iter_rhs) {
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
R_API void r_table_uniq(RTable *t) {
|
|
r_table_group (t, -1, NULL);
|
|
}
|
|
|
|
R_API void r_table_group(RTable *t, int nth, RTableSelector fcn) {
|
|
RListIter *iter;
|
|
RListIter *tmp;
|
|
RTableRow *row;
|
|
|
|
RListIter *iter_inner;
|
|
RTableRow *uniq_row;
|
|
|
|
RList *rows = t->rows;
|
|
|
|
r_list_foreach_safe (rows, iter, tmp, row) {
|
|
for (iter_inner = rows->head;
|
|
iter_inner && iter_inner != iter;
|
|
iter_inner = iter_inner->n) {
|
|
|
|
uniq_row = iter_inner->data;
|
|
|
|
if (!r_rows_cmp (uniq_row->items, row->items, t->cols, nth)) {
|
|
if (fcn) {
|
|
fcn (uniq_row, row, nth);
|
|
}
|
|
r_list_delete (rows, iter);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
R_API int r_table_column_nth(RTable *t, const char *name) {
|
|
RListIter *iter;
|
|
RTableColumn *col;
|
|
int n = 0;
|
|
r_list_foreach (t->cols, iter, col) {
|
|
if (!strcmp (name, col->name)) {
|
|
return n;
|
|
}
|
|
n++;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
static int __resolveOperation(const char *op) {
|
|
if (!strcmp (op, "gt")) {
|
|
return '>';
|
|
}
|
|
if (!strcmp (op, "lt")) {
|
|
return '<';
|
|
}
|
|
if (!strcmp (op, "eq")) {
|
|
return '=';
|
|
}
|
|
if (!strcmp (op, "ne")) {
|
|
return '!';
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
static void __table_column_free(void *_col) {
|
|
RTableColumn *col = (RTableColumn*)_col;
|
|
free (col);
|
|
}
|
|
|
|
R_API void r_table_columns(RTable *t, RList *col_names) {
|
|
// 1 bool per OLD column to indicate whether it should be freed (masked out)
|
|
bool *free_cols = malloc (sizeof (bool) * r_list_length (t->cols));
|
|
if (!free_cols) {
|
|
return;
|
|
}
|
|
size_t i;
|
|
for (i = 0; i < r_list_length (t->cols); i++) {
|
|
free_cols[i] = true;
|
|
}
|
|
|
|
// 1 value per NEW column to indicate from which OLD column to take the info from and whether to dup it
|
|
struct col_source {
|
|
int oldcol;
|
|
bool dup;
|
|
} *col_sources = calloc (r_list_length (col_names), sizeof (struct col_source));
|
|
if (!col_sources) {
|
|
free (free_cols);
|
|
return;
|
|
}
|
|
|
|
// First create the plan which new columns to take from which old, which ones to dup or free.
|
|
RListIter *it;
|
|
const char *col_name;
|
|
size_t new_count = 0;
|
|
r_list_foreach (col_names, it, col_name) {
|
|
int fc = r_table_column_nth (t, col_name);
|
|
if (fc < 0) {
|
|
continue;
|
|
}
|
|
col_sources[new_count].oldcol = fc;
|
|
col_sources[new_count].dup = !free_cols[fc]; // if we already used the same old column for another new column before, we must dup it for all following!
|
|
free_cols[fc] = false;
|
|
new_count++;
|
|
}
|
|
|
|
RTableRow *row;
|
|
r_list_foreach (t->rows, it, row) {
|
|
RList *old_items = row->items;
|
|
RList *new_items = r_list_newf (free);
|
|
for (i = 0; i < new_count; i++) {
|
|
char *item = r_list_get_n (old_items, col_sources[i].oldcol);
|
|
if (!item) {
|
|
continue;
|
|
}
|
|
if (col_sources[i].dup) {
|
|
item = strdup (item);
|
|
}
|
|
r_list_append (new_items, item);
|
|
}
|
|
row->items = new_items;
|
|
|
|
// Free dropped items
|
|
char *item;
|
|
i = 0;
|
|
RListIter *fit;
|
|
r_list_foreach (old_items, fit, item) {
|
|
if (free_cols[i]) {
|
|
free (item);
|
|
}
|
|
i++;
|
|
}
|
|
old_items->free = NULL;
|
|
r_list_free (old_items);
|
|
}
|
|
|
|
RList *old_cols = t->cols;
|
|
RList *new_cols = r_list_newf (r_table_column_free);
|
|
for (i = 0; i < new_count; i++) {
|
|
RTableColumn *col = r_list_get_n (old_cols, col_sources[i].oldcol);
|
|
if (!col) {
|
|
continue;
|
|
}
|
|
if (col_sources[i].dup) {
|
|
col = r_table_column_clone (col);
|
|
}
|
|
r_list_append (new_cols, col);
|
|
}
|
|
t->cols = new_cols;
|
|
|
|
// Free dropped columns
|
|
RTableColumn *col;
|
|
i = 0;
|
|
r_list_foreach (old_cols, it, col) {
|
|
if (free_cols[i]) {
|
|
r_table_column_free (col);
|
|
}
|
|
i++;
|
|
}
|
|
old_cols->free = NULL;
|
|
r_list_free (old_cols);
|
|
|
|
free (col_sources);
|
|
free (free_cols);
|
|
}
|
|
|
|
R_API void r_table_filter_columns(RTable *t, RList *list) {
|
|
const char *col;
|
|
RListIter *iter;
|
|
RList *cols = t->cols;
|
|
t->cols = r_list_newf (__table_column_free);
|
|
r_list_foreach (list, iter, col) {
|
|
int ncol = r_table_column_nth (t, col);
|
|
if (ncol != -1) {
|
|
RTableColumn *c = r_list_get_n (cols, ncol);
|
|
if (c) {
|
|
r_table_add_column (t, c->type, col, 0);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
R_API const char *r_table_help(void) {
|
|
return \
|
|
"RTableQuery> comma separated. 'c' stands for column name.\n"
|
|
" c/sort/inc sort rows by given colname\n"
|
|
" c/sortlen/inc sort rows by strlen()\n"
|
|
" c/cols/c1/c2 only show selected columns\n"
|
|
" c/gt/0x800 grep rows matching col0 > 0x800\n"
|
|
" c/lt/0x800 grep rows matching col0 < 0x800\n"
|
|
" c/eq/0x800 grep rows matching col0 == 0x800\n"
|
|
" c/ne/0x800 grep rows matching col0 != 0x800\n"
|
|
" */uniq get the first row of each that col0 is unique\n"
|
|
" */head/10 same as | head -n 10\n"
|
|
" */skip/10 skip the first 10 rows\n"
|
|
" */tail/10 same as | tail -n 10\n"
|
|
" */page/1/10 show the first 10 rows (/page/2/10 will show the 2nd)\n"
|
|
" c/str/warn grep rows matching col(name).str(warn)\n"
|
|
" c/nostr/warn grep rows not matching col(name).str(warn)\n"
|
|
" c/strlen/3 grep rows matching strlen(col) == X\n"
|
|
" c/minlen/3 grep rows matching strlen(col) > X\n"
|
|
" c/maxlen/3 grep rows matching strlen(col) < X\n"
|
|
" c/sum sum all the values of given column\n"
|
|
" :r2 .tostring() == .tor2() # supports import/export\n"
|
|
" :csv .tostring() == .tocsv() # supports import/export\n"
|
|
" :tsv .tostring() == .totsv() # supports import/export\n"
|
|
" :fancy .tostring() == .tofancystring()\n"
|
|
" :html .tostring() == .tohtml()\n"
|
|
" :json .tostring() == .tojson()\n"
|
|
" :simple simple table output without lines\n"
|
|
" :sql .tostring() == .tosql() # export table contents in SQL statements\n"
|
|
" :header show column headers (see :quiet and :noheader)\n"
|
|
" :quiet do not print column names header\n";
|
|
}
|
|
|
|
static bool __table_special(RTable *t, const char *columnName) {
|
|
if (*columnName != ':') {
|
|
return false;
|
|
}
|
|
if (!strcmp (columnName, ":quiet")) {
|
|
t->showHeader = false;
|
|
t->showFancy = false;
|
|
} else if (r_str_startswith (columnName, ":nohead")) {
|
|
t->showHeader = false;
|
|
} else if (r_str_startswith (columnName, ":head")) {
|
|
t->showHeader = true;
|
|
} else if (!strcmp (columnName, ":fancy")) {
|
|
t->showHeader = true;
|
|
t->showFancy = true;
|
|
} else if (!strcmp (columnName, ":sql")) {
|
|
t->showSQL = true;
|
|
t->showHeader = false;
|
|
} else if (!strcmp (columnName, ":simple")) {
|
|
t->showHeader = true;
|
|
t->showFancy = false;
|
|
} else if (!strcmp (columnName, ":r2")) {
|
|
t->showR2 = true;
|
|
} else if (!strcmp (columnName, ":csv")) {
|
|
t->showCSV = true;
|
|
} else if (!strcmp (columnName, ":html")) {
|
|
t->showHTML = true;
|
|
} else if (!strcmp (columnName, ":tsv")) {
|
|
t->showTSV = true;
|
|
} else if (!strcmp (columnName, ":json")) {
|
|
t->showJSON = true;
|
|
} else {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
R_API bool r_table_query(RTable *t, const char *q) {
|
|
r_return_val_if_fail (t, false);
|
|
q = r_str_trim_head_ro (q);
|
|
// TODO support parenthesis and (or)||
|
|
// split by "&&" (or comma) -> run .filter on each
|
|
// addr/gt/200,addr/lt/400,addr/sort/dec,offset/sort/inc
|
|
if (!q || !*q) {
|
|
__table_adjust (t);
|
|
return true;
|
|
}
|
|
if (*q == '?' || r_str_endswith (q, ":help")) {
|
|
const char *th = r_table_help ();
|
|
eprintf ("%s\n", th);
|
|
return false;
|
|
}
|
|
|
|
RListIter *iter;
|
|
char *qq = strdup (q);
|
|
RList *queries = r_str_split_list (qq, ",", 0);
|
|
char *query;
|
|
r_list_foreach (queries, iter, query) {
|
|
RList *q = r_str_split_list (query, "/", 2);
|
|
const char *columnName = r_list_get_n (q, 0);
|
|
const char *operation = r_list_get_n (q, 1);
|
|
const char *operand = r_list_get_n (q, 2);
|
|
if (__table_special (t, columnName)) {
|
|
r_list_free (q);
|
|
continue;
|
|
}
|
|
int col = r_table_column_nth (t, columnName);
|
|
if (col == -1) {
|
|
if (columnName == NULL && strcmp (operation, "uniq")) {
|
|
R_LOG_ERROR ("Invalid column name (%s) for (%s)", columnName, query);
|
|
} else if (columnName) {
|
|
if (*columnName == '[') {
|
|
col = atoi (columnName + 1);
|
|
}
|
|
}
|
|
}
|
|
if (!operation) {
|
|
r_list_free (q);
|
|
break;
|
|
}
|
|
if (!strcmp (operation, "sort")) {
|
|
r_table_sort (t, col, operand && !strcmp (operand, "dec"));
|
|
} else if (!strcmp (operation, "uniq")) {
|
|
r_table_group (t, col, NULL);
|
|
} else if (!strcmp (operation, "sortlen")) {
|
|
r_table_sortlen (t, col, operand && !strcmp (operand, "dec"));
|
|
} else if (!strcmp (operation, "join")) {
|
|
// TODO: implement join operation with other command's tables
|
|
} else if (!strcmp (operation, "sum")) {
|
|
char *op = strdup (r_str_get (operand));
|
|
RList *list = r_str_split_list (op, "/", 0);
|
|
r_list_prepend (list, strdup (columnName));
|
|
r_table_columns (t, list); // select/reorder columns
|
|
r_list_free (list);
|
|
r_table_filter (t, 0, '+', op);
|
|
free (op);
|
|
} else if (!strcmp (operation, "strlen")) {
|
|
if (operand) {
|
|
r_table_filter (t, col, 's', operand);
|
|
}
|
|
} else if (!strcmp (operation, "minlen")) {
|
|
if (operand) {
|
|
r_table_filter (t, col, 'l', operand);
|
|
}
|
|
} else if (!strcmp (operation, "maxlen")) {
|
|
if (operand) {
|
|
r_table_filter (t, col, 'L', operand);
|
|
}
|
|
} else if (!strcmp (operation, "page")) {
|
|
if (operand) {
|
|
r_table_filter (t, col, 'p', operand);
|
|
}
|
|
} else if (!strcmp (operation, "tail")) {
|
|
if (operand) {
|
|
r_table_filter (t, col, 't', operand);
|
|
}
|
|
} else if (!strcmp (operation, "skip")) {
|
|
if (operand) {
|
|
r_table_filter (t, col, 'S', operand);
|
|
}
|
|
} else if (!strcmp (operation, "head")) {
|
|
if (operand) {
|
|
r_table_filter (t, col, 'h', operand);
|
|
}
|
|
} else if (!strcmp (operation, "nostr")) {
|
|
if (operand) {
|
|
r_table_filter (t, col, '$', operand);
|
|
}
|
|
} else if (!strcmp (operation, "str")) {
|
|
if (operand) {
|
|
r_table_filter (t, col, '~', operand);
|
|
}
|
|
} else if (!strcmp (operation, "cols")) {
|
|
char *op = strdup (r_str_get (operand));
|
|
RList *list = r_str_split_list (op, "/", 0);
|
|
r_list_prepend (list, (char *)columnName);
|
|
r_table_columns (t, list); // select/reorder columns
|
|
r_list_free (list);
|
|
free (op);
|
|
// TODO r_table_filter_columns (t, q);
|
|
} else {
|
|
int op = __resolveOperation (operation);
|
|
if (op == -1) {
|
|
R_LOG_ERROR ("Invalid table operation (%s)", operation);
|
|
} else {
|
|
r_table_filter (t, col, op, operand);
|
|
}
|
|
}
|
|
r_list_free (q);
|
|
}
|
|
r_list_free (queries);
|
|
free (qq);
|
|
__table_adjust (t);
|
|
return true;
|
|
}
|
|
|
|
R_API bool r_table_align(RTable *t, int nth, int align) {
|
|
RTableColumn *col = r_list_get_n (t->cols, nth);
|
|
if (col) {
|
|
col->align = align;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
R_API void r_table_hide_header(RTable *t) {
|
|
t->showHeader = false;
|
|
}
|
|
|
|
R_API void r_table_visual_list(RTable *table, RList *list, ut64 seek, ut64 len, int width, bool va) {
|
|
ut64 mul, min = -1, max = -1;
|
|
RListIter *iter;
|
|
RListInfo *info;
|
|
RCons *cons = (RCons *) table->cons;
|
|
table->showHeader = false;
|
|
const char *h_line = cons->use_utf8 ? RUNE_LONG_LINE_HORIZ : "-";
|
|
const char *block = cons->use_utf8 ? R_UTF8_BLOCK : "#";
|
|
int j, i;
|
|
width -= 80;
|
|
if (width < 1) {
|
|
width = 30;
|
|
}
|
|
|
|
r_table_set_columnsf (table, "sssssss", "No.", "offset", "blocks", "offset", "perms", "extra", "name");
|
|
r_list_foreach (list, iter, info) {
|
|
if (min == -1 || info->pitv.addr < min) {
|
|
min = info->pitv.addr;
|
|
}
|
|
if (max == -1 || info->pitv.addr + info->pitv.size > max) {
|
|
max = info->pitv.addr + info->pitv.size;
|
|
}
|
|
}
|
|
mul = (max - min) / width;
|
|
if (min != -1 && mul > 0) {
|
|
i = 0;
|
|
r_list_foreach (list, iter, info) {
|
|
RStrBuf *buf = r_strbuf_new ("");
|
|
for (j = 0; j < width; j++) {
|
|
ut64 pos = min + j * mul;
|
|
ut64 npos = min + (j + 1) * mul;
|
|
const char *arg = (info->pitv.addr < npos && (info->pitv.addr + info->pitv.size) > pos)
|
|
? block: h_line;
|
|
r_strbuf_append (buf, arg);
|
|
}
|
|
r_strf_var (a0, 64, "%d%c", i, (va
|
|
? r_itv_contain (info->vitv, seek)
|
|
: r_itv_contain (info->pitv, seek))? '*' : ' ');
|
|
r_strf_var (a1, 64, "0x%08"PFMT64x, va
|
|
? info->vitv.addr: info->pitv.addr);
|
|
r_strf_var (a2, 64, "0x%08"PFMT64x, va
|
|
? r_itv_end (info->vitv): r_itv_end (info->pitv));
|
|
char *b = r_strbuf_drain (buf);
|
|
r_table_add_rowf (table, "sssssss",
|
|
a0, a1, b, a2,
|
|
(info->perm != -1)? r_str_rwx_i (info->perm) : "",
|
|
(info->extra)?info->extra : "",
|
|
(info->name)?info->name :"");
|
|
free (b);
|
|
i++;
|
|
}
|
|
/* current seek */
|
|
if (i > 0 && len != 0) {
|
|
RStrBuf *buf = r_strbuf_new ("");
|
|
if (seek == UT64_MAX) {
|
|
seek = 0;
|
|
}
|
|
for (j = 0; j < width; j++) {
|
|
r_strbuf_append (buf,((j * mul) + min >= seek &&
|
|
(j * mul) + min <= seek + len) ? "^" : h_line);
|
|
}
|
|
r_strf_var (a0, 64, "0x%08"PFMT64x, seek);
|
|
r_strf_var (a1, 64, "0x%08"PFMT64x, seek + len);
|
|
char *b = r_strbuf_drain (buf);
|
|
r_table_add_rowf (table, "sssssss", "=>", a0, b, a1, "", "", "");
|
|
free (b);
|
|
}
|
|
}
|
|
}
|
|
|
|
R_API RTable *r_table_clone(const RTable *t) {
|
|
RTable *o = r_table_new (t->name);
|
|
RTableColumn *col;
|
|
RTableRow *row;
|
|
RListIter *iter;
|
|
r_list_foreach (t->cols, iter, col) {
|
|
r_list_append (o->rows, r_table_column_clone (col));
|
|
}
|
|
r_list_foreach (t->rows, iter, row) {
|
|
r_list_append (o->rows, r_table_row_clone (row));
|
|
}
|
|
return o;
|
|
}
|
|
|
|
#if 0
|
|
// TODO: to be implemented
|
|
|
|
R_API RTable *r_table_push(RTable *t) {
|
|
// TODO: implement
|
|
return NULL;
|
|
}
|
|
|
|
R_API RTable *r_table_pop(RTable *t) {
|
|
// TODO: implement
|
|
return NULL;
|
|
}
|
|
|
|
R_API void r_table_fromjson(RTable *t, const char *csv) {
|
|
// TODO
|
|
}
|
|
|
|
R_API void r_table_fromtsv(RTable *t, const char *csv) {
|
|
}
|
|
|
|
R_API void r_table_fromcsv(RTable *t, const char *csv) {
|
|
// TODO
|
|
}
|
|
|
|
R_API void r_table_transpose(RTable *t) {
|
|
// When the music stops rows will be cols and cols... rows!
|
|
}
|
|
|
|
R_API void r_table_format(RTable *t, int nth, RTableColumnType *type) {
|
|
// change the format of a specific column
|
|
// change imm base, decimal precission, ...
|
|
}
|
|
|
|
// to compute sum result of all the elements in a column
|
|
R_API ut64 r_table_reduce(RTable *t, int nth) {
|
|
// When the music stops rows will be cols and cols... rows!
|
|
return 0;
|
|
}
|
|
#endif
|