mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-10 07:21:55 +00:00
620 lines
16 KiB
C
620 lines
16 KiB
C
/* radare - LGPL - Copyright 2009-2018 - pancake */
|
|
|
|
#include "r_cons.h"
|
|
#include "r_core.h"
|
|
#include "r_types.h"
|
|
#include "r_io.h"
|
|
|
|
static const char *help_msg_S[] = {
|
|
"Usage:","S[?-.*=adlr] [...]","",
|
|
"S","","list sections",
|
|
"S"," paddr va sz [vsz] name rwx","add new section (if(!vsz)vsz=sz)",
|
|
"S.","","show current section name",
|
|
"S.-*","","remove all sections in current offset",
|
|
"S*","","list sections (in radare commands)",
|
|
"S-[id]","","remove section identified by id",
|
|
"S-.","","remove section at core->offset (can be changed with @)",
|
|
"S=","","list sections (ascii-art bars) (io.va to display paddr or vaddr)",
|
|
"Sa","[-] [A] [B] [[off]]","Specify arch and bits for given section",
|
|
"Sd[a]"," [file]","dump current (all) section to a file (see dmd)",
|
|
"Sf"," [baddr]","Alias for S 0 0 $s $s foo rwx",
|
|
"Sj","","list sections in JSON (alias for iSj)",
|
|
"Sl"," [file]","load contents of file into current section (see dml)",
|
|
"Sr"," [name]","rename section on current seek",
|
|
"SR", "[?]", "Remap sections with different mode of operation",
|
|
NULL
|
|
};
|
|
|
|
static const char *help_msg_Sl[] = {
|
|
"Usage:", "Sl", "[file]",
|
|
NULL
|
|
};
|
|
|
|
static const char *help_msg_Sr[] = {
|
|
"Usage:", "Sr", "[name] ([offset])",
|
|
NULL
|
|
};
|
|
|
|
static const char* help_msg_SR[] = {
|
|
"Usage:","SR[b|s][a|p|e] [id]","",
|
|
"SRb", "[a|p|e] binid", "Remap sections of binid for Analysis, Patch or Emulation",
|
|
"SRs","[a|p|e] secid","Remap section with sectid for Analysis, Patch or Emulation",
|
|
NULL
|
|
};
|
|
|
|
static void cmd_section_init(RCore *core) {
|
|
DEFINE_CMD_DESCRIPTOR (core, S);
|
|
DEFINE_CMD_DESCRIPTOR (core, Sl);
|
|
DEFINE_CMD_DESCRIPTOR (core, Sr);
|
|
}
|
|
|
|
#define PRINT_CURRENT_SEEK \
|
|
if (i > 0 && len != 0) { \
|
|
if (seek == UT64_MAX) seek = 0; \
|
|
io->cb_printf ("=> 0x%08"PFMT64x" |", seek); \
|
|
for (j = 0; j < width; j++) { \
|
|
io->cb_printf ( \
|
|
((j*mul) + min >= seek && \
|
|
(j*mul) + min <= seek + len) \
|
|
? "^" : "-"); \
|
|
} \
|
|
io->cb_printf ("| 0x%08"PFMT64x"\n", seek+len); \
|
|
}
|
|
|
|
static void list_section_visual(RIO *io, ut64 seek, ut64 len, int use_color, int cols) {
|
|
ut64 mul, min = -1, max = -1;
|
|
SdbListIter *iter;
|
|
RIOSection *s;
|
|
int j, i = 0;
|
|
int width = cols - 70;
|
|
if (width < 1) {
|
|
width = 30;
|
|
}
|
|
// seek = r_io_section_vaddr_to_maddr_try (io, seek);
|
|
// seek = r_io_section_vaddr_to_maddr_try (io, seek);
|
|
ls_foreach (io->sections, iter, s) {
|
|
if (min == -1 || s->paddr < min) {
|
|
min = s->paddr;
|
|
}
|
|
if (max == -1 || s->paddr+s->size > max) {
|
|
max = s->paddr + s->size;
|
|
}
|
|
}
|
|
mul = (max-min) / width;
|
|
if (min != -1 && mul != 0) {
|
|
const char * color = "", *color_end = "";
|
|
char buf[128];
|
|
i = 0;
|
|
ls_foreach (io->sections, iter, s) {
|
|
r_num_units (buf, s->size);
|
|
if (use_color) {
|
|
color_end = Color_RESET;
|
|
if (s->flags & 1) { // exec bit
|
|
color = Color_GREEN;
|
|
} else if (s->flags & 2) { // write bit
|
|
color = Color_RED;
|
|
} else {
|
|
color = "";
|
|
color_end = "";
|
|
}
|
|
} else {
|
|
color = "";
|
|
color_end = "";
|
|
}
|
|
if (io->va) {
|
|
io->cb_printf ("%02d%c %s0x%08"PFMT64x"%s |", s->id,
|
|
(seek >= s->vaddr && seek < s->vaddr + s->vsize) ? '*' : ' ',
|
|
color, s->vaddr, color_end);
|
|
} else {
|
|
io->cb_printf ("%02d%c %s0x%08"PFMT64x"%s |", s->id,
|
|
(seek >= s->paddr && seek < s->paddr + s->size) ? '*' : ' ',
|
|
color, s->paddr, color_end);
|
|
}
|
|
for (j = 0; j < width; j++) {
|
|
ut64 pos = min + (j * mul);
|
|
ut64 npos = min + ((j + 1) * mul);
|
|
if (s->paddr < npos && (s->paddr + s->size) > pos)
|
|
io->cb_printf ("#");
|
|
else io->cb_printf ("-");
|
|
}
|
|
if (io->va) {
|
|
io->cb_printf ("| %s0x%08"PFMT64x"%s %5s %s %04s\n",
|
|
color, s->vaddr + s->vsize, color_end, buf,
|
|
r_str_rwx_i (s->flags), s->name);
|
|
} else {
|
|
io->cb_printf ("| %s0x%08"PFMT64x"%s %5s %s %04s\n",
|
|
color, s->paddr+s->size, color_end, buf,
|
|
r_str_rwx_i (s->flags), s->name);
|
|
}
|
|
|
|
i++;
|
|
}
|
|
PRINT_CURRENT_SEEK;
|
|
}
|
|
}
|
|
|
|
static void __section_list (RIO *io, ut64 offset, RPrint *print, int rad) {
|
|
SdbListIter *iter;
|
|
RIOSection *s;
|
|
|
|
if (!io || !io->sections || !print || !print->cb_printf) {
|
|
return;
|
|
}
|
|
if (rad == '=') { // "S="
|
|
int cols = r_cons_get_size (NULL);
|
|
list_section_visual (io, offset, -1, print->flags & R_PRINT_FLAGS_COLOR, cols);
|
|
} else if (rad) {
|
|
ls_foreach (io->sections, iter, s) {
|
|
char *n = strdup (s->name);
|
|
r_name_filter (n, strlen (n));
|
|
print->cb_printf ("f section.%s %"PFMT64d" 0x%"PFMT64x"\n", n, s->size, s->vaddr);
|
|
print->cb_printf ("S 0x%08"PFMT64x" 0x%08"PFMT64x" 0x%08"
|
|
PFMT64x" 0x%08"PFMT64x" %s %s\n", s->paddr,
|
|
s->vaddr, s->size, s->vsize, n, r_str_rwx_i (s->flags));
|
|
free (n);
|
|
}
|
|
} else {
|
|
ls_foreach (io->sections, iter, s) {
|
|
print->cb_printf ("[%02d:%02d]", s->bin_id, s->id);
|
|
if (io->va) {
|
|
if ((s->vaddr <= offset) && ((offset - s->vaddr) < s->vsize)) {
|
|
print->cb_printf (" * ");
|
|
} else {
|
|
print->cb_printf (" . ");
|
|
}
|
|
} else {
|
|
if ((s->paddr <= offset) && ((offset - s->paddr) < s->size)) {
|
|
print->cb_printf (" * ");
|
|
} else {
|
|
print->cb_printf (" . ");
|
|
}
|
|
}
|
|
print->cb_printf ("pa=0x%08"PFMT64x" %s va=0x%08"PFMT64x
|
|
" sz=0x%04"PFMT64x" vsz=0x%04"PFMT64x" %s", s->paddr,
|
|
r_str_rwx_i (s->flags), s->vaddr, s->size, s->vsize, s->name);
|
|
if (s->arch && s->bits) {
|
|
print->cb_printf (" ; %s %d", r_sys_arch_str (s->arch),
|
|
s->bits);
|
|
}
|
|
print->cb_printf ("\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
static bool dumpSectionsToDisk(RCore *core) {
|
|
char file[512];
|
|
SdbListIter *iter;
|
|
RIOSection *s;
|
|
|
|
ls_foreach (core->io->sections, iter, s) {
|
|
ut8 *buf = malloc (s->size);
|
|
r_io_read_at (core->io, s->paddr, buf, s->size);
|
|
snprintf (file, sizeof (file),
|
|
"0x%08"PFMT64x"-0x%08"PFMT64x"-%s.dmp",
|
|
s->vaddr, s->vaddr+s->size,
|
|
r_str_rwx_i (s->flags));
|
|
if (!r_file_dump (file, buf, s->size, 0)) {
|
|
eprintf ("Cannot write '%s'\n", file);
|
|
free (buf);
|
|
return false;
|
|
}
|
|
eprintf ("Dumped %d byte(s) into %s\n", (int)s->size, file);
|
|
free (buf);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static bool dumpSectionToDisk(RCore *core, char *file) {
|
|
char *heapfile = NULL;
|
|
SdbListIter *iter;
|
|
RIOSection *s;
|
|
int len = 128;
|
|
if (!core || !file) {
|
|
return false;
|
|
}
|
|
ut64 o = core->offset;
|
|
if (core->io->va || core->io->debug) {
|
|
s = r_io_section_vget (core->io, o);
|
|
o = s ? o - s->vaddr + s->paddr : o;
|
|
}
|
|
ls_foreach (core->io->sections, iter, s) {
|
|
if (o >= s->paddr && o < s->paddr + s->size) {
|
|
ut8 *buf = malloc (s->size);
|
|
r_io_read_at (core->io, s->paddr, buf, s->size);
|
|
if (!file) {
|
|
heapfile = (char *)malloc (len);
|
|
if (!heapfile) {
|
|
free (buf);
|
|
return false;
|
|
}
|
|
file = heapfile;
|
|
snprintf (file, len,
|
|
"0x%08"PFMT64x"-0x%08"PFMT64x"-%s.dmp",
|
|
s->vaddr, s->vaddr + s->size,
|
|
r_str_rwx_i (s->flags));
|
|
}
|
|
if (!r_file_dump (file, buf, s->size, 0)) {
|
|
eprintf ("Cannot write '%s'\n", file);
|
|
free (buf);
|
|
free (heapfile);
|
|
return false;
|
|
}
|
|
eprintf ("Dumped %d byte(s) into %s\n", (int)s->size, file);
|
|
free (buf);
|
|
free (heapfile);
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static void update_section_flag_at_with_oldname(RIOSection *s, RFlag *flags, ut64 off, char *oldname) {
|
|
RFlagItem *item = NULL;
|
|
RListIter *iter;
|
|
const RList *list = NULL;
|
|
int len = 0;
|
|
char *secname = NULL;
|
|
list = r_flag_get_list (flags, s->vaddr);
|
|
secname = sdb_fmt ("section.%s", oldname);
|
|
len = strlen (secname);
|
|
r_list_foreach (list, iter, item) {
|
|
if (!item->name) {
|
|
continue;
|
|
}
|
|
if (!strncmp (item->name, secname, R_MIN (strlen (item->name), len))) {
|
|
free (item->realname);
|
|
item->name = strdup (sdb_fmt ("section.%s", s->name));
|
|
r_str_trim (item->name);
|
|
r_name_filter (item->name, 0);
|
|
item->realname = item->name;
|
|
break;
|
|
}
|
|
}
|
|
list = r_flag_get_list (flags, s->vaddr + s->size);
|
|
secname = sdb_fmt ("section_end.%s", oldname);
|
|
len = strlen (secname);
|
|
r_list_foreach (list, iter, item) {
|
|
if (!item->name) {
|
|
continue;
|
|
}
|
|
if (!strncmp (item->name, secname, R_MIN (strlen (item->name), len))) {
|
|
free (item->realname);
|
|
item->name = strdup (sdb_fmt ("section_end.%s", s->name));
|
|
r_str_trim (item->name);
|
|
r_name_filter (item->name, 0);
|
|
item->realname = item->name;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static int cmd_section_reapply(RCore *core, const char *input) {
|
|
int mode = 0;
|
|
ut32 id;
|
|
switch (*input) {
|
|
case '?':
|
|
r_core_cmd_help (core, help_msg_SR);
|
|
break;
|
|
case 'b':
|
|
case 's':
|
|
switch (input[1]) {
|
|
case 'e':
|
|
mode = R_IO_SECTION_APPLY_FOR_EMULATOR;
|
|
break;
|
|
case 'p':
|
|
mode = R_IO_SECTION_APPLY_FOR_PATCH;
|
|
break;
|
|
case 'a':
|
|
mode = R_IO_SECTION_APPLY_FOR_ANALYSIS;
|
|
break;
|
|
default:
|
|
r_core_cmd_help (core, help_msg_SR);
|
|
return 0;
|
|
}
|
|
if (*input == 'b') {
|
|
id = (ut32)r_num_math (core->num, input + 2);
|
|
if (!r_io_section_reapply_bin (core->io, id, mode)) {
|
|
eprintf ("Cannot reapply section with binid %d\n", id);
|
|
}
|
|
} else {
|
|
id = (ut32)r_num_math (core->num, input + 2);
|
|
if (!r_io_section_reapply (core->io, id, mode)) {
|
|
eprintf ("Cannot reapply section with secid %d\n", id);
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
r_core_cmd_help (core, help_msg_SR);
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_section(void *data, const char *input) {
|
|
RCore *core = (RCore *)data;
|
|
switch (*input) {
|
|
case '?': // "S?"
|
|
r_core_cmd_help (core, help_msg_S);
|
|
// TODO: add command to resize current section
|
|
break;
|
|
case 'R' : // "SR"
|
|
return cmd_section_reapply (core, input + 1);
|
|
case 'f': // "Sf"
|
|
if (input[1] == ' ') {
|
|
ut64 n = r_num_math (core->num, input + 1);
|
|
r_core_cmdf (core, "S 0x%"PFMT64x" 0x%"PFMT64x" $s $s foo rwx", n, n);
|
|
} else {
|
|
r_core_cmd0 (core, "S 0 0 $s $s foo rwx");
|
|
}
|
|
break;
|
|
case 'j': // "Sj"
|
|
r_core_cmd0 (core, "iSj");
|
|
break;
|
|
case 'a': // "Sa"
|
|
switch (input[1]) {
|
|
case '\0':
|
|
{
|
|
int b = 0;
|
|
const char *n = r_io_section_get_archbits (core->io, core->offset, &b);
|
|
if (n) {
|
|
r_cons_printf ("%s %d\n", n, b);
|
|
}
|
|
}
|
|
break;
|
|
case '-': // "Sa-"
|
|
r_io_section_set_archbits (core->io, core->offset, NULL, 0);
|
|
break;
|
|
case '?': // "Sa?"
|
|
default:
|
|
eprintf ("Usage: Sa[-][arch] [bits] [[off]]\n");
|
|
break;
|
|
case ' ': // "Sa "
|
|
{
|
|
int i;
|
|
char *ptr = strdup (input+2);
|
|
const char *arch = NULL;
|
|
char bits = 0;
|
|
ut64 offset = core->offset;
|
|
i = r_str_word_set0 (ptr);
|
|
if (i < 2) {
|
|
eprintf ("Missing argument\n");
|
|
free (ptr);
|
|
break;
|
|
}
|
|
if (i == 3) {
|
|
offset = r_num_math (core->num, r_str_word_get0 (ptr, 2));
|
|
}
|
|
bits = r_num_math (core->num, r_str_word_get0 (ptr, 1));
|
|
arch = r_str_word_get0 (ptr, 0);
|
|
if (r_io_section_set_archbits (core->io, offset, arch, bits)) {
|
|
core->section = NULL;
|
|
r_core_seek (core, core->offset, 0);
|
|
} else {
|
|
eprintf ("Cannot set arch/bits at 0x%08"PFMT64x"\n",offset);
|
|
}
|
|
free (ptr);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case 'r': // "Sr"
|
|
if (input[1] == ' ') {
|
|
RIOSection *s;
|
|
// int len = 0;
|
|
ut64 vaddr;
|
|
char *p = strchr (input + 2, ' ');
|
|
if (p) {
|
|
*p = 0;
|
|
vaddr = r_num_math (core->num, p + 1);
|
|
// len = (int)(size_t)(p-input + 2);
|
|
} else {
|
|
vaddr = core->offset;
|
|
}
|
|
s = r_io_section_vget (core->io, vaddr);
|
|
if (s) {
|
|
char *oldname = s->name;
|
|
s->name = strdup (input + 2);
|
|
//update flag space for the given section
|
|
update_section_flag_at_with_oldname (s, core->flags, s->vaddr, oldname);
|
|
free (oldname);
|
|
} else {
|
|
eprintf ("No section found in 0x%08"PFMT64x"\n", core->offset);
|
|
}
|
|
} else {
|
|
r_core_cmd_help (core, help_msg_Sr);
|
|
}
|
|
break;
|
|
case 'd': // "Sd"
|
|
{
|
|
char *file = NULL;
|
|
int len = 128;
|
|
switch (input[1]) {
|
|
case 0:
|
|
(void) dumpSectionToDisk (core, NULL);
|
|
break;
|
|
case ' ': // "Sd "
|
|
if (input[2]) {
|
|
file = (char *)calloc (len, sizeof (char));
|
|
if (file) {
|
|
snprintf (file, len, "%s", input + 2);
|
|
}
|
|
}
|
|
(void) dumpSectionToDisk (core, file);
|
|
free (file);
|
|
break;
|
|
case 'a': // "Sda"
|
|
(void)dumpSectionsToDisk (core);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case 'l': // "Sl"
|
|
{
|
|
ut64 o = core->offset;
|
|
SdbListIter *iter;
|
|
RIOSection *s;
|
|
if (input[1] != ' ') {
|
|
r_core_cmd_help (core, help_msg_Sl);
|
|
return false;
|
|
}
|
|
if (core->io->va || core->io->debug) {
|
|
s = r_io_section_vget (core->io, core->offset);
|
|
o = s ? core->offset - s->vaddr + s->paddr : core->offset;
|
|
}
|
|
ls_foreach (core->io->sections, iter, s) {
|
|
if (o >= s->paddr && o < s->paddr + s->size) {
|
|
int sz;
|
|
char *buf = r_file_slurp (input + 2, &sz);
|
|
// TODO: use mmap here. we need a portable implementation
|
|
if (!buf) {
|
|
eprintf ("Cannot allocate 0x%08"PFMT64x""
|
|
" bytes\n", s->size);
|
|
return false;
|
|
}
|
|
r_io_write_at (core->io, s->vaddr, (const ut8*)buf, sz);
|
|
eprintf ("Loaded %d byte(s) into the map region "
|
|
" at 0x%08"PFMT64x"\n", sz, s->vaddr);
|
|
free (buf);
|
|
return true;
|
|
}
|
|
}
|
|
eprintf ("No debug region found here\n");
|
|
return false;
|
|
}
|
|
break;
|
|
case '-': // "S-"
|
|
// remove all sections
|
|
if (input[1] == '*') {
|
|
r_io_section_init (core->io);
|
|
}
|
|
if (input[1] == '.') {
|
|
RIOSection *s = r_io_section_vget (core->io, core->offset);
|
|
if (!s) {
|
|
return 0;
|
|
}
|
|
// use offset
|
|
r_io_section_rm (core->io, s->id);
|
|
}
|
|
if (input[1]) {
|
|
r_io_section_rm (core->io, atoi (input + 1));
|
|
}
|
|
break;
|
|
case ' ': // "S "
|
|
switch (input[1]) {
|
|
case '-': // "S -" remove
|
|
if (input[2] == '?' || input[2] == '\0') {
|
|
eprintf ("Usage: S -N # where N is the "
|
|
" section index\n");
|
|
} else {
|
|
r_io_section_rm (core->io, atoi (input + 1));
|
|
}
|
|
break;
|
|
default:
|
|
{
|
|
int i, rwx = 7;
|
|
char *ptr = strdup (input + 1);
|
|
const char *name = NULL;
|
|
char vname[64];
|
|
ut64 vaddr = 0LL;
|
|
ut64 paddr = 0LL;
|
|
ut64 size = 0LL;
|
|
ut64 vsize = 0LL;
|
|
int bin_id = -1;
|
|
int fd = r_core_file_cur_fd (core);
|
|
i = r_str_word_set0 (ptr);
|
|
switch (i) {
|
|
case 7: //get bin id
|
|
bin_id = r_num_math (core->num, r_str_word_get0 (ptr, 6));
|
|
case 6: // get rwx
|
|
rwx = r_str_rwx (r_str_word_get0 (ptr, 5));
|
|
case 5: // get name
|
|
name = r_str_word_get0 (ptr, 4);
|
|
case 4: // get vsize
|
|
vsize = r_num_math (core->num, r_str_word_get0 (ptr, 3));
|
|
case 3: // get size
|
|
size = r_num_math (core->num, r_str_word_get0 (ptr, 2));
|
|
case 2: // get vaddr
|
|
vaddr = r_num_math (core->num, r_str_word_get0 (ptr, 1));
|
|
case 1: // get paddr
|
|
paddr = r_num_math (core->num, r_str_word_get0 (ptr, 0));
|
|
}
|
|
if (!vsize) {
|
|
vsize = size;
|
|
if (i > 3) {
|
|
name = r_str_word_get0 (ptr, 3);
|
|
}
|
|
if (i > 4) {
|
|
rwx = r_str_rwx (r_str_word_get0 (ptr, 4));
|
|
}
|
|
}
|
|
if (!name || !*name) {
|
|
sprintf (vname, "area%d", (int)ls_length (core->io->sections));
|
|
name = vname;
|
|
}
|
|
RIOSection *sec = r_io_section_add (core->io, paddr, vaddr, size, vsize, rwx, name, bin_id, fd);
|
|
r_io_create_mem_for_section (core->io, sec);
|
|
free (ptr);
|
|
}
|
|
break;
|
|
}
|
|
break;
|
|
case '.': // "S."
|
|
if (input[1] == '-') { // "S.-"
|
|
ut64 o = core->offset;
|
|
SdbListIter *iter, *iter2;
|
|
RIOSection *s;
|
|
if (core->io->va || core->io->debug) {
|
|
s = r_io_section_vget (core->io, o);
|
|
o = s ? o - s->vaddr + s->paddr : o;
|
|
}
|
|
ls_foreach_safe (core->io->sections, iter, iter2, s) {
|
|
if (o >= s->paddr && o < s->paddr + s->size) {
|
|
r_io_section_rm (core->io, s->id);
|
|
if (input[2] != '*') {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
ut64 o = core->offset;
|
|
SdbListIter *iter;
|
|
RIOSection *s;
|
|
if (core->io->va || core->io->debug) {
|
|
s = r_io_section_vget (core->io, o);
|
|
o = s ? o - s->vaddr + s->paddr : o;
|
|
}
|
|
if (input[1] == 'j') { // "S.j"
|
|
r_cons_printf ("[");
|
|
ls_foreach (core->io->sections, iter, s) {
|
|
if (o >= s->paddr && o < s->paddr + s->size) {
|
|
char *name = r_str_escape (s->name);
|
|
r_cons_printf ("{\"start\":%" PFMT64u ",\"end\":%" PFMT64u ",\"name\":\"%s\"}",
|
|
s->paddr + s->vaddr,
|
|
s->paddr + s->vaddr + s->size,
|
|
name);
|
|
free (name);
|
|
}
|
|
}
|
|
r_cons_printf ("]");
|
|
} else {
|
|
ls_foreach (core->io->sections, iter, s) {
|
|
if (o >= s->paddr && o < s->paddr + s->size) {
|
|
r_cons_printf ("0x%08"PFMT64x" 0x%08"PFMT64x" %s\n",
|
|
s->paddr + s->vaddr,
|
|
s->paddr + s->vaddr + s->size,
|
|
s->name);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case '\0': // "S"
|
|
case '=': // "S="
|
|
case '*': // "S*"
|
|
__section_list (core->io, core->offset, core->print, *input);
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|