2010-05-28 17:15:20 +02:00
|
|
|
/* radare - LGPL - Copyright 2009-2010 nibble<.ds@gmail.com> */
|
|
|
|
|
|
|
|
/* TODO:
|
|
|
|
* -L [lib] dlopen library and show address
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <getopt.h>
|
|
|
|
|
|
|
|
#include <r_types.h>
|
|
|
|
#include <r_list.h>
|
|
|
|
#include <r_lib.h>
|
|
|
|
#include <r_bin.h>
|
|
|
|
#include <r_flags.h>
|
|
|
|
#include <r_util.h>
|
|
|
|
|
|
|
|
#define ACTION_UNK 0x0000
|
2010-07-29 16:04:18 +02:00
|
|
|
#define ACTION_ENTRIES 0x0001
|
|
|
|
#define ACTION_IMPORTS 0x0002
|
|
|
|
#define ACTION_SYMBOLS 0x0004
|
|
|
|
#define ACTION_SECTIONS 0x0008
|
2010-05-28 17:15:20 +02:00
|
|
|
#define ACTION_INFO 0x0010
|
|
|
|
#define ACTION_OPERATION 0x0020
|
|
|
|
#define ACTION_HELP 0x0040
|
2010-07-29 16:04:18 +02:00
|
|
|
#define ACTION_STRINGS 0x0080
|
|
|
|
#define ACTION_FIELDS 0x0100
|
|
|
|
#define ACTION_LIBS 0x0200
|
|
|
|
#define ACTION_SRCLINE 0x0400
|
|
|
|
#define ACTION_MAIN 0x0800
|
|
|
|
#define ACTION_EXTRACT 0x1000
|
2010-09-10 11:11:38 +02:00
|
|
|
#define ACTION_RELOCS 0x2000
|
2010-09-24 21:23:13 +02:00
|
|
|
#define ACTION_LISTARCHS 0x4000
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
static struct r_lib_t *l;
|
|
|
|
static struct r_bin_t *bin = NULL;
|
|
|
|
static int rad = R_FALSE;
|
|
|
|
static int rw = R_FALSE;
|
|
|
|
static int va = R_FALSE;
|
2010-08-16 14:35:15 +02:00
|
|
|
static ut64 gbaddr = 0LL;
|
2010-05-28 17:15:20 +02:00
|
|
|
static char* file = NULL;
|
2010-10-04 13:57:48 +02:00
|
|
|
static char* output = "out";
|
2010-08-17 13:15:18 +02:00
|
|
|
static ut64 at = 0LL;
|
|
|
|
static char *name = NULL;
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
static int rabin_show_help() {
|
|
|
|
printf ("rabin2 [options] [file]\n"
|
2010-10-01 10:09:50 +02:00
|
|
|
" -A List archs\n"
|
2010-10-04 03:46:58 +02:00
|
|
|
" -a [arch_bits] Set arch\n"
|
|
|
|
" -f [str] Select sub-bin named str\n"
|
2010-10-01 10:09:50 +02:00
|
|
|
" -b [addr] Override baddr\n"
|
|
|
|
" -e Entrypoint\n"
|
|
|
|
" -M Main\n"
|
|
|
|
" -i Imports (symbols imported from libraries)\n"
|
|
|
|
" -s Symbols (exports)\n"
|
|
|
|
" -S Sections\n"
|
|
|
|
" -z Strings\n"
|
|
|
|
" -I Binary info\n"
|
|
|
|
" -H Header fields\n"
|
|
|
|
" -l Linked libraries\n"
|
|
|
|
" -R Relocations\n"
|
|
|
|
" -O [str] Write/Extract operations (str=help for help)\n"
|
2010-10-04 13:57:48 +02:00
|
|
|
" -o [str] Output file/folder for write operations (out by default)\n"
|
2010-10-01 10:09:50 +02:00
|
|
|
" -r radare output\n"
|
|
|
|
" -v Use vaddr in radare output\n"
|
|
|
|
" -m [addr] Show source line at addr\n"
|
|
|
|
" -L List supported bin plugins\n"
|
|
|
|
" -@ [addr] Show section, symbol or import at addr\n"
|
|
|
|
" -n [str] Show section, symbol or import named str\n"
|
|
|
|
" -x Extract bins contained in file\n"
|
|
|
|
" -V Show version information\n"
|
|
|
|
" -h This help\n");
|
2010-09-24 21:23:13 +02:00
|
|
|
return 1;
|
2010-05-28 17:15:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static int rabin_show_entrypoints() {
|
|
|
|
RList *entries;
|
|
|
|
RListIter *iter;
|
2010-05-30 06:06:25 +02:00
|
|
|
RBinAddr *entry;
|
2010-05-28 17:15:20 +02:00
|
|
|
int i = 0;
|
|
|
|
|
2010-08-16 14:35:15 +02:00
|
|
|
ut64 baddr = gbaddr?gbaddr:r_bin_get_baddr (bin);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
if ((entries = r_bin_get_entries (bin)) == NULL)
|
|
|
|
return R_FALSE;
|
|
|
|
|
|
|
|
if (rad) printf ("fs symbols\n");
|
2010-09-24 21:23:13 +02:00
|
|
|
else eprintf ("[Entrypoints]\n");
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
r_list_foreach (entries, iter, entry) {
|
|
|
|
if (rad) {
|
|
|
|
printf ("f entry%i @ 0x%08"PFMT64x"\n", i, va?baddr+entry->rva:entry->offset);
|
|
|
|
printf ("s entry%i\n", i);
|
|
|
|
} else printf ("address=0x%08"PFMT64x" offset=0x%08"PFMT64x" baddr=0x%08"PFMT64x"\n",
|
|
|
|
baddr+entry->rva, entry->offset, baddr);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
2010-09-24 21:23:13 +02:00
|
|
|
if (!rad) eprintf ("\n%i entrypoints\n", i);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-05-30 06:06:25 +02:00
|
|
|
static int rabin_show_main() {
|
|
|
|
RBinAddr *binmain;
|
2010-08-16 14:35:15 +02:00
|
|
|
ut64 baddr = gbaddr?gbaddr:r_bin_get_baddr (bin);
|
2010-05-30 06:06:25 +02:00
|
|
|
|
|
|
|
if ((binmain = r_bin_get_main (bin)) == NULL)
|
|
|
|
return R_FALSE;
|
|
|
|
|
|
|
|
if (rad) printf ("fs symbols\n");
|
2010-09-24 21:23:13 +02:00
|
|
|
else eprintf ("[Main]\n");
|
2010-05-30 06:06:25 +02:00
|
|
|
|
|
|
|
if (rad) {
|
|
|
|
printf ("f main @ 0x%08"PFMT64x"\n", va?baddr+binmain->rva:binmain->offset);
|
|
|
|
} else printf ("address=0x%08"PFMT64x" offset=0x%08"PFMT64x"\n",
|
|
|
|
baddr+binmain->rva, binmain->offset);
|
|
|
|
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-09-25 12:33:30 +02:00
|
|
|
static int rabin_extract(int all) {
|
2010-10-04 13:57:48 +02:00
|
|
|
char outfile[512], outpath[512], *path, *ptr;
|
2010-10-01 11:10:59 +02:00
|
|
|
int i = 0;
|
2010-10-01 08:12:43 +02:00
|
|
|
|
2010-10-04 13:57:48 +02:00
|
|
|
// XXX: Wrong for w32 (/)
|
2010-09-25 12:33:30 +02:00
|
|
|
if (all) {
|
|
|
|
for (i=0; i<bin->narch; i++) {
|
2010-10-04 04:01:25 +02:00
|
|
|
r_bin_set_archidx (bin, i);
|
|
|
|
if (bin->curarch.info == NULL) {
|
2010-10-01 11:58:11 +02:00
|
|
|
eprintf ("No extract info found.\n");
|
|
|
|
} else {
|
2010-10-04 13:57:48 +02:00
|
|
|
path = strdup (bin->curarch.file);
|
|
|
|
if ((ptr = strrchr (path, '/'))) {
|
|
|
|
*ptr = '\0';
|
2010-10-01 11:58:11 +02:00
|
|
|
ptr = ptr+1;
|
2010-10-04 13:57:48 +02:00
|
|
|
}
|
2010-10-04 04:01:25 +02:00
|
|
|
else ptr = bin->curarch.file;
|
2010-10-04 13:57:48 +02:00
|
|
|
snprintf (outpath, sizeof (outpath), "%s/%s", output, path);
|
|
|
|
if (!r_sys_rmkdir (outpath)) {
|
|
|
|
eprintf ("Error creating dir structure\n");
|
2010-10-01 11:58:11 +02:00
|
|
|
return R_FALSE;
|
2010-10-04 13:57:48 +02:00
|
|
|
}
|
|
|
|
snprintf (outfile, sizeof (outfile), "%s/%s.%s_%i",
|
|
|
|
outpath, ptr, bin->curarch.info->arch,
|
|
|
|
bin->curarch.info->bits);
|
|
|
|
if (!r_file_dump (outfile, bin->curarch.buf->buf, bin->curarch.size)) {
|
|
|
|
eprintf ("Error extracting %s\n", outfile);
|
|
|
|
return R_FALSE;
|
|
|
|
} else printf ("%s created (%i)\n", outfile, bin->curarch.size);
|
2010-10-01 11:58:11 +02:00
|
|
|
}
|
2010-09-25 12:33:30 +02:00
|
|
|
}
|
2010-10-04 13:57:48 +02:00
|
|
|
} else { /* XXX: Use 'output' for filename? */
|
2010-10-04 04:01:25 +02:00
|
|
|
if (bin->curarch.info == NULL) {
|
2010-10-01 11:10:59 +02:00
|
|
|
eprintf ("No extract info found.\n");
|
|
|
|
} else {
|
2010-10-04 04:01:25 +02:00
|
|
|
if ((ptr = strrchr (bin->curarch.file, '/')))
|
2010-10-01 08:12:43 +02:00
|
|
|
ptr = ptr+1;
|
2010-10-04 04:01:25 +02:00
|
|
|
else ptr = bin->curarch.file;
|
2010-10-04 13:57:48 +02:00
|
|
|
snprintf (outfile, sizeof (outfile), "%s.%s_%i", ptr,
|
2010-10-04 04:01:25 +02:00
|
|
|
bin->curarch.info->arch, bin->curarch.info->bits);
|
2010-10-04 13:57:48 +02:00
|
|
|
if (!r_file_dump (outfile, bin->curarch.buf->buf, bin->curarch.size)) {
|
|
|
|
eprintf ("Error extracting %s\n", outfile);
|
2010-09-25 12:33:30 +02:00
|
|
|
return R_FALSE;
|
2010-10-04 13:57:48 +02:00
|
|
|
} else printf ("%s created (%i)\n", outfile, bin->curarch.size);
|
2010-09-25 12:33:30 +02:00
|
|
|
}
|
|
|
|
}
|
2010-09-24 21:23:13 +02:00
|
|
|
return R_TRUE;
|
2010-07-29 16:04:18 +02:00
|
|
|
}
|
|
|
|
|
2010-05-28 17:15:20 +02:00
|
|
|
static int rabin_show_libs() {
|
|
|
|
RList *libs;
|
|
|
|
RListIter *iter;
|
|
|
|
char* lib;
|
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
if ((libs = r_bin_get_libs (bin)) == NULL)
|
|
|
|
return R_FALSE;
|
|
|
|
|
2010-09-24 21:23:13 +02:00
|
|
|
eprintf ("[Linked libraries]\n");
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
r_list_foreach (libs, iter, lib) {
|
|
|
|
printf ("%s\n", lib);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
2010-09-24 21:23:13 +02:00
|
|
|
if (!rad) eprintf ("\n%i libraries\n", i);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-09-10 11:11:38 +02:00
|
|
|
static int rabin_show_relocs() {
|
|
|
|
RList *relocs;
|
|
|
|
RListIter *iter;
|
|
|
|
RBinReloc *reloc;
|
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
ut64 baddr = gbaddr?gbaddr:r_bin_get_baddr (bin);
|
|
|
|
|
|
|
|
if ((relocs = r_bin_get_relocs (bin)) == NULL)
|
|
|
|
return R_FALSE;
|
|
|
|
|
|
|
|
if (rad) printf ("fs relocs\n");
|
2010-09-24 21:23:13 +02:00
|
|
|
else eprintf ("[Relocations]\n");
|
2010-09-10 11:11:38 +02:00
|
|
|
|
|
|
|
r_list_foreach (relocs, iter, reloc) {
|
2010-10-04 10:55:43 +02:00
|
|
|
if (rad) printf ("f reloc.%s @ 0x%08"PFMT64x"\n", reloc->name, va?baddr+reloc->rva:reloc->offset);
|
|
|
|
else printf ("sym=%02i address=0x%08"PFMT64x" offset=0x%08"PFMT64x" type=0x%08x %s\n",
|
2010-09-10 11:11:38 +02:00
|
|
|
reloc->sym, baddr+reloc->rva, reloc->offset, reloc->type, reloc->name);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
2010-09-24 21:23:13 +02:00
|
|
|
if (!rad) eprintf ("\n%i relocations\n", i);
|
2010-09-10 11:11:38 +02:00
|
|
|
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-08-17 13:15:18 +02:00
|
|
|
static int rabin_show_imports() {
|
2010-05-28 17:15:20 +02:00
|
|
|
RList *imports;
|
|
|
|
RListIter *iter;
|
|
|
|
RBinImport *import;
|
|
|
|
ut64 baddr;
|
|
|
|
int i = 0;
|
|
|
|
|
2010-08-16 14:35:15 +02:00
|
|
|
baddr = gbaddr?gbaddr:r_bin_get_baddr (bin);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
if ((imports = r_bin_get_imports (bin)) == NULL)
|
|
|
|
return R_FALSE;
|
|
|
|
|
|
|
|
if (!at && !rad)
|
2010-09-24 21:23:13 +02:00
|
|
|
eprintf ("[Imports]\n");
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
r_list_foreach (imports, iter, import) {
|
2010-08-17 13:15:18 +02:00
|
|
|
if (name && strcmp (import->name, name))
|
|
|
|
continue;
|
2010-05-28 17:15:20 +02:00
|
|
|
if (at) {
|
|
|
|
if (baddr+import->rva == at || import->offset == at)
|
|
|
|
printf ("%s\n", import->name);
|
|
|
|
} else {
|
|
|
|
if (rad) {
|
|
|
|
r_flag_name_filter (import->name);
|
2010-08-01 11:02:55 +02:00
|
|
|
if (import->size)
|
|
|
|
printf ("af+ 0x%08"PFMT64x" %"PFMT64d" fcn.imp.%s\n",
|
|
|
|
va?baddr+import->rva:import->offset, import->size, import->name);
|
2010-05-28 17:15:20 +02:00
|
|
|
printf ("fs imports\n");
|
|
|
|
printf ("f imp.%s @ 0x%08"PFMT64x"\n",
|
|
|
|
import->name, va?baddr+import->rva:import->offset);
|
|
|
|
printf ("fs functions\n");
|
|
|
|
printf ("f fcn.imp.%s @ 0x%08"PFMT64x"\n",
|
|
|
|
import->name, va?baddr+import->rva:import->offset);
|
|
|
|
} else printf ("address=0x%08"PFMT64x" offset=0x%08"PFMT64x" ordinal=%03"PFMT64d" "
|
|
|
|
"hint=%03"PFMT64d" bind=%s type=%s name=%s\n",
|
|
|
|
baddr+import->rva, import->offset,
|
|
|
|
import->ordinal, import->hint, import->bind,
|
|
|
|
import->type, import->name);
|
|
|
|
}
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
2010-09-24 21:23:13 +02:00
|
|
|
if (!at && !rad) eprintf ("\n%i imports\n", i);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-08-17 13:15:18 +02:00
|
|
|
static int rabin_show_symbols() {
|
2010-05-28 17:15:20 +02:00
|
|
|
RList *symbols;
|
|
|
|
RListIter *iter;
|
|
|
|
RBinSymbol *symbol;
|
|
|
|
ut64 baddr;
|
|
|
|
int i = 0;
|
|
|
|
|
2010-08-16 14:35:15 +02:00
|
|
|
baddr = gbaddr?gbaddr:r_bin_get_baddr (bin);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
if ((symbols = r_bin_get_symbols (bin)) == NULL)
|
|
|
|
return R_FALSE;
|
|
|
|
|
|
|
|
if (!at) {
|
|
|
|
if (rad) printf ("fs symbols\n");
|
2010-09-24 21:23:13 +02:00
|
|
|
else eprintf ("[Symbols]\n");
|
2010-05-28 17:15:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
r_list_foreach (symbols, iter, symbol) {
|
2010-08-17 13:15:18 +02:00
|
|
|
if (name && strcmp (symbol->name, name))
|
|
|
|
continue;
|
2010-05-28 17:15:20 +02:00
|
|
|
if (at) {
|
|
|
|
if ((symbol->size != 0 &&
|
|
|
|
((baddr+symbol->rva <= at && baddr+symbol->rva+symbol->size > at) ||
|
|
|
|
(symbol->offset <= at && symbol->offset+symbol->size > at))) ||
|
|
|
|
baddr+symbol->rva == at || symbol->offset == at)
|
|
|
|
printf("%s\n", symbol->name);
|
|
|
|
} else {
|
|
|
|
if (rad) {
|
|
|
|
r_flag_name_filter (symbol->name);
|
|
|
|
if (!strncmp (symbol->type,"FUNC", 4)) {
|
|
|
|
printf ("fs functions\n");
|
|
|
|
printf ("f fcn.sym.%s %"PFMT64d" 0x%08"PFMT64x"\n",
|
|
|
|
symbol->name, symbol->size,
|
|
|
|
va?baddr+symbol->rva:symbol->offset);
|
|
|
|
printf ("fs symbols\n");
|
|
|
|
} else if (!strncmp (symbol->type,"OBJECT", 6))
|
|
|
|
printf ("Cd %"PFMT64d" @ 0x%08"PFMT64x"\n",
|
|
|
|
symbol->size, va?baddr+symbol->rva:symbol->offset);
|
|
|
|
printf ("f sym.%s %"PFMT64d" 0x%08"PFMT64x"\n",
|
|
|
|
symbol->name, symbol->size,
|
|
|
|
va?baddr+symbol->rva:symbol->offset);
|
|
|
|
} else printf ("address=0x%08"PFMT64x" offset=0x%08"PFMT64x" ordinal=%03"PFMT64d" "
|
|
|
|
"forwarder=%s size=%"PFMT64d" bind=%s type=%s name=%s\n",
|
|
|
|
baddr+symbol->rva, symbol->offset,
|
|
|
|
symbol->ordinal, symbol->forwarder,
|
|
|
|
symbol->size, symbol->bind, symbol->type,
|
|
|
|
symbol->name);
|
|
|
|
}
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
2010-09-24 21:23:13 +02:00
|
|
|
if (!at && !rad) eprintf ("\n%i symbols\n", i);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int rabin_show_strings() {
|
|
|
|
RList *strings;
|
|
|
|
RListIter *iter;
|
|
|
|
RBinString *string;
|
|
|
|
RBinSection *section;
|
|
|
|
int i = 0;
|
2010-08-16 14:35:15 +02:00
|
|
|
ut64 baddr = gbaddr?gbaddr:r_bin_get_baddr (bin);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
if ((strings = r_bin_get_strings (bin)) == NULL)
|
|
|
|
return R_FALSE;
|
|
|
|
|
|
|
|
if (rad) printf ("fs strings\n");
|
2010-09-24 21:23:13 +02:00
|
|
|
else eprintf ("[strings]\n");
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
r_list_foreach (strings, iter, string) {
|
|
|
|
section = r_bin_get_section_at (bin, string->offset, 0);
|
|
|
|
if (rad) {
|
|
|
|
r_flag_name_filter (string->string);
|
|
|
|
printf ("f str.%s %"PFMT64d" @ 0x%08"PFMT64x"\n"
|
2010-08-15 21:30:59 +02:00
|
|
|
"Cs %"PFMT64d" @ 0x%08"PFMT64x"\n",
|
|
|
|
string->string, string->size, va?baddr+string->rva:string->offset,
|
|
|
|
string->size, va?baddr+string->rva:string->offset);
|
2010-05-28 17:15:20 +02:00
|
|
|
} else printf ("address=0x%08"PFMT64x" offset=0x%08"PFMT64x" ordinal=%03"PFMT64d" "
|
2010-08-15 21:30:59 +02:00
|
|
|
"size=%"PFMT64d" section=%s string=%s\n",
|
|
|
|
baddr+string->rva, string->offset,
|
|
|
|
string->ordinal, string->size,
|
|
|
|
section?section->name:"unknown", string->string);
|
2010-05-28 17:15:20 +02:00
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
2010-09-24 21:23:13 +02:00
|
|
|
if (!rad) eprintf ("\n%i strings\n", i);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-08-17 13:15:18 +02:00
|
|
|
static int rabin_show_sections() {
|
2010-05-28 17:15:20 +02:00
|
|
|
RList *sections;
|
|
|
|
RListIter *iter;
|
|
|
|
RBinSection *section;
|
|
|
|
ut64 baddr;
|
|
|
|
int i = 0;
|
|
|
|
|
2010-08-16 14:35:15 +02:00
|
|
|
baddr = gbaddr?gbaddr:r_bin_get_baddr (bin);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
if ((sections = r_bin_get_sections (bin)) == NULL)
|
|
|
|
return R_FALSE;
|
|
|
|
|
|
|
|
if (!at) {
|
|
|
|
if (rad) printf ("fs sections\n");
|
2010-09-24 21:23:13 +02:00
|
|
|
else eprintf ("[Sections]\n");
|
2010-05-28 17:15:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
r_list_foreach (sections, iter, section) {
|
2010-08-17 13:15:18 +02:00
|
|
|
if (name && strcmp (section->name, name))
|
|
|
|
continue;
|
2010-05-28 17:15:20 +02:00
|
|
|
if (at) {
|
|
|
|
if ((section->size != 0 &&
|
|
|
|
((baddr+section->rva <= at && baddr+section->rva+section->size > at) ||
|
|
|
|
(section->offset <= at && section->offset+section->size > at))) ||
|
|
|
|
baddr+section->rva == at || section->offset == at)
|
|
|
|
printf ("%s\n", section->name);
|
|
|
|
} else {
|
|
|
|
if (rad) {
|
|
|
|
r_flag_name_filter (section->name);
|
|
|
|
printf ("S 0x%08"PFMT64x" 0x%08"PFMT64x" 0x%08"PFMT64x" 0x%08"PFMT64x" %s %d\n",
|
2010-08-15 21:30:59 +02:00
|
|
|
section->offset, baddr+section->rva,
|
|
|
|
section->size, section->vsize, section->name, (int)section->srwx);
|
2010-05-28 17:15:20 +02:00
|
|
|
printf ("f section.%s %"PFMT64d" 0x%08"PFMT64x"\n",
|
2010-08-15 21:30:59 +02:00
|
|
|
section->name, section->size, va?baddr+section->rva:section->offset);
|
2010-05-28 17:15:20 +02:00
|
|
|
printf ("CC [%02i] va=0x%08"PFMT64x" pa=0x%08"PFMT64x" sz=%"PFMT64d" vsz=%"PFMT64d" "
|
2010-08-15 21:30:59 +02:00
|
|
|
"rwx=%c%c%c%c %s @ 0x%08"PFMT64x"\n",
|
|
|
|
i, baddr+section->rva, section->offset, section->size, section->vsize,
|
|
|
|
R_BIN_SCN_SHAREABLE (section->srwx)?'s':'-',
|
|
|
|
R_BIN_SCN_READABLE (section->srwx)?'r':'-',
|
|
|
|
R_BIN_SCN_WRITABLE (section->srwx)?'w':'-',
|
|
|
|
R_BIN_SCN_EXECUTABLE (section->srwx)?'x':'-',
|
|
|
|
section->name,va?baddr+section->rva:section->offset);
|
2010-05-28 17:15:20 +02:00
|
|
|
} else printf ("idx=%02i address=0x%08"PFMT64x" offset=0x%08"PFMT64x" size=%"PFMT64d" vsize=%"PFMT64d" "
|
2010-08-15 21:30:59 +02:00
|
|
|
"privileges=%c%c%c%c name=%s\n",
|
|
|
|
i, baddr+section->rva, section->offset, section->size, section->vsize,
|
|
|
|
R_BIN_SCN_SHAREABLE (section->srwx)?'s':'-',
|
|
|
|
R_BIN_SCN_READABLE (section->srwx)?'r':'-',
|
|
|
|
R_BIN_SCN_WRITABLE (section->srwx)?'w':'-',
|
|
|
|
R_BIN_SCN_EXECUTABLE (section->srwx)?'x':'-',
|
|
|
|
section->name);
|
2010-05-28 17:15:20 +02:00
|
|
|
}
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
2010-09-24 21:23:13 +02:00
|
|
|
if (!at && !rad) eprintf ("\n%i sections\n", i);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int rabin_show_info() {
|
|
|
|
RBinInfo *info;
|
|
|
|
|
|
|
|
if ((info = r_bin_get_info (bin)) == NULL)
|
|
|
|
return R_FALSE;
|
|
|
|
|
|
|
|
if (rad) {
|
|
|
|
printf ("e file.type=%s\n"
|
|
|
|
"e cfg.bigendian=%s\n"
|
|
|
|
"e asm.os=%s\n"
|
|
|
|
"e asm.arch=%s\n"
|
2010-09-28 13:58:03 +02:00
|
|
|
"e anal.plugin=%s\n"
|
2010-05-28 17:15:20 +02:00
|
|
|
"e asm.bits=%i\n"
|
|
|
|
"e asm.dwarf=%s\n",
|
2010-09-28 13:58:03 +02:00
|
|
|
info->rclass, info->big_endian?"true":"false", info->os,
|
|
|
|
info->arch, info->arch, info->bits,
|
|
|
|
R_BIN_DBG_STRIPPED (info->dbg_info)?"false":"true");
|
2010-09-24 21:23:13 +02:00
|
|
|
} else {
|
|
|
|
eprintf ("[File info]\n");
|
|
|
|
printf ("File=%s\n"
|
|
|
|
"Type=%s\n"
|
|
|
|
"Class=%s\n"
|
|
|
|
"Arch=%s %i\n"
|
|
|
|
"Machine=%s\n"
|
|
|
|
"OS=%s\n"
|
|
|
|
"Subsystem=%s\n"
|
|
|
|
"Big endian=%s\n"
|
|
|
|
"Stripped=%s\n"
|
|
|
|
"Static=%s\n"
|
|
|
|
"Line_nums=%s\n"
|
|
|
|
"Local_syms=%s\n"
|
|
|
|
"Relocs=%s\n"
|
|
|
|
"RPath=%s\n",
|
|
|
|
info->file, info->type, info->bclass,
|
|
|
|
info->arch, info->bits, info->machine, info->os,
|
|
|
|
info->subsystem, info->big_endian?"True":"False",
|
|
|
|
R_BIN_DBG_STRIPPED (info->dbg_info)?"True":"False",
|
|
|
|
R_BIN_DBG_STATIC (info->dbg_info)?"True":"False",
|
|
|
|
R_BIN_DBG_LINENUMS (info->dbg_info)?"True":"False",
|
|
|
|
R_BIN_DBG_SYMS (info->dbg_info)?"True":"False",
|
|
|
|
R_BIN_DBG_RELOCS (info->dbg_info)?"True":"False",
|
|
|
|
info->rpath);
|
|
|
|
}
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int rabin_show_fields() {
|
|
|
|
RList *fields;
|
|
|
|
RListIter *iter;
|
|
|
|
RBinField *field;
|
|
|
|
ut64 baddr;
|
|
|
|
int i = 0;
|
|
|
|
|
2010-08-16 14:35:15 +02:00
|
|
|
baddr = gbaddr?gbaddr:r_bin_get_baddr (bin);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
if ((fields = r_bin_get_fields (bin)) == NULL)
|
|
|
|
return R_FALSE;
|
|
|
|
|
|
|
|
if (rad) printf ("fs header\n");
|
2010-09-24 21:23:13 +02:00
|
|
|
else eprintf ("[Header fields]\n");
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
r_list_foreach (fields, iter, field) {
|
|
|
|
if (rad) {
|
|
|
|
r_flag_name_filter (field->name);
|
|
|
|
printf ("f header.%s @ 0x%08"PFMT64x"\n",
|
|
|
|
field->name, va?baddr+field->rva:field->offset);
|
|
|
|
printf ("[%02i] address=0x%08"PFMT64x" offset=0x%08"PFMT64x" name=%s\n",
|
|
|
|
i, baddr+field->rva, field->offset, field->name);
|
|
|
|
} else printf ("idx=%02i address=0x%08"PFMT64x" offset=0x%08"PFMT64x" name=%s\n",
|
|
|
|
i, baddr+field->rva, field->offset, field->name);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
2010-09-24 21:23:13 +02:00
|
|
|
if (!rad) eprintf ("\n%i fields\n", i);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int rabin_dump_symbols(int len) {
|
|
|
|
RList *symbols;
|
|
|
|
RListIter *iter;
|
|
|
|
RBinSymbol *symbol;
|
|
|
|
ut8 *buf;
|
|
|
|
char *ret;
|
|
|
|
int olen = len;
|
|
|
|
|
|
|
|
if ((symbols = r_bin_get_symbols (bin)) == NULL)
|
|
|
|
return R_FALSE;
|
|
|
|
|
|
|
|
r_list_foreach (symbols, iter, symbol) {
|
|
|
|
if (symbol->size != 0 && (olen > symbol->size || olen == 0))
|
|
|
|
len = symbol->size;
|
|
|
|
else if (symbol->size == 0 && olen == 0)
|
|
|
|
len = 32;
|
|
|
|
else len = olen;
|
|
|
|
|
|
|
|
if (!(buf = malloc (len)) || !(ret = malloc(len*2+1)))
|
|
|
|
return R_FALSE;
|
2010-10-04 03:46:58 +02:00
|
|
|
r_buf_read_at (bin->curarch.buf, symbol->offset, buf, len);
|
2010-05-28 17:15:20 +02:00
|
|
|
r_hex_bin2str (buf, len, ret);
|
|
|
|
printf ("%s %s\n", symbol->name, ret);
|
|
|
|
free (buf);
|
|
|
|
free (ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-08-17 13:15:18 +02:00
|
|
|
static int rabin_dump_sections(char *scnname) {
|
2010-05-28 17:15:20 +02:00
|
|
|
RList *sections;
|
|
|
|
RListIter *iter;
|
|
|
|
RBinSection *section;
|
|
|
|
ut8 *buf;
|
|
|
|
char *ret;
|
|
|
|
|
|
|
|
if ((sections = r_bin_get_sections (bin)) == NULL)
|
|
|
|
return R_FALSE;
|
|
|
|
|
|
|
|
r_list_foreach (sections, iter, section) {
|
2010-08-17 13:15:18 +02:00
|
|
|
if (!strcmp (scnname, section->name)) {
|
2010-05-28 17:15:20 +02:00
|
|
|
if (!(buf = malloc (section->size)) ||
|
|
|
|
!(ret = malloc (section->size*2+1)))
|
|
|
|
return R_FALSE;
|
2010-10-04 03:46:58 +02:00
|
|
|
r_buf_read_at (bin->curarch.buf, section->offset, buf, section->size);
|
2010-05-28 17:15:20 +02:00
|
|
|
r_hex_bin2str (buf, section->size, ret);
|
|
|
|
printf ("%s\n", ret);
|
|
|
|
free (buf);
|
|
|
|
free (ret);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int rabin_do_operation(const char *op) {
|
|
|
|
char *arg = NULL, *ptr = NULL, *ptr2 = NULL;
|
|
|
|
|
|
|
|
if (!strcmp (op, "help")) {
|
|
|
|
printf ("Operation string:\n"
|
|
|
|
" Dump symbols: d/s/1024\n"
|
|
|
|
" Dump section: d/S/.text\n"
|
|
|
|
" Resize section: r/.data/1024\n");
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
/* Implement alloca with fixed-size buffer? */
|
|
|
|
if (!(arg = strdup (op)))
|
|
|
|
return R_FALSE;
|
|
|
|
|
|
|
|
if ((ptr = strchr (arg, '/'))) {
|
|
|
|
ptr[0] = '\0';
|
|
|
|
ptr = ptr + 1;
|
|
|
|
if ((ptr2 = strchr (ptr, '/'))) {
|
|
|
|
ptr2[0] = '\0';
|
|
|
|
ptr2 = ptr2 + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (arg[0]) {
|
|
|
|
case 'd':
|
|
|
|
if (!ptr)
|
|
|
|
goto _rabin_do_operation_error;
|
|
|
|
if (ptr[0]=='s') {
|
|
|
|
if (ptr2) {
|
|
|
|
if (!rabin_dump_symbols (r_num_math(NULL, ptr2)))
|
|
|
|
return R_FALSE;
|
|
|
|
} else
|
|
|
|
if (!rabin_dump_symbols (0))
|
|
|
|
return R_FALSE;
|
|
|
|
} else if (ptr[0]=='S') {
|
|
|
|
if (!ptr2)
|
|
|
|
goto _rabin_do_operation_error;
|
|
|
|
if (!rabin_dump_sections (ptr2))
|
|
|
|
return R_FALSE;
|
|
|
|
} else goto _rabin_do_operation_error;
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
r_bin_wr_scn_resize (bin, ptr, r_num_math (NULL, ptr2));
|
|
|
|
r_bin_wr_output (bin, output);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
_rabin_do_operation_error:
|
2010-09-24 21:23:13 +02:00
|
|
|
eprintf ("Unknown operation. use -O help\n");
|
2010-05-28 17:15:20 +02:00
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
free (arg);
|
|
|
|
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-08-17 13:15:18 +02:00
|
|
|
static int rabin_show_srcline() {
|
2010-05-28 17:15:20 +02:00
|
|
|
char *srcline;
|
|
|
|
if ((srcline = r_bin_meta_get_source_line (bin, at))) {
|
|
|
|
printf ("%s\n", srcline);
|
|
|
|
free (srcline);
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* bin callback */
|
|
|
|
static int __lib_bin_cb(struct r_lib_plugin_t *pl, void *user, void *data) {
|
|
|
|
struct r_bin_plugin_t *hand = (struct r_bin_plugin_t *)data;
|
|
|
|
//printf(" * Added (dis)assembly plugin\n");
|
|
|
|
r_bin_add (bin, hand);
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __lib_bin_dt(struct r_lib_plugin_t *pl, void *p, void *u) {
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-09-24 21:23:13 +02:00
|
|
|
/* binxtr callback */
|
|
|
|
static int __lib_bin_xtr_cb(struct r_lib_plugin_t *pl, void *user, void *data) {
|
|
|
|
struct r_bin_xtr_plugin_t *hand = (struct r_bin_xtr_plugin_t *)data;
|
|
|
|
//printf(" * Added (dis)assembly plugin\n");
|
|
|
|
r_bin_xtr_add (bin, hand);
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __lib_bin_xtr_dt(struct r_lib_plugin_t *pl, void *p, void *u) {
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-10-04 00:42:11 +02:00
|
|
|
int main(int argc, char **argv) {
|
2010-10-01 10:09:50 +02:00
|
|
|
int c, bits = 0;
|
2010-05-28 17:15:20 +02:00
|
|
|
int action = ACTION_UNK;
|
2010-09-25 03:45:03 +02:00
|
|
|
const char *op = NULL;
|
2010-10-04 03:46:58 +02:00
|
|
|
char *arch = NULL, *arch_name = NULL;
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
bin = r_bin_new ();
|
|
|
|
l = r_lib_new ("radare_plugin");
|
|
|
|
r_lib_add_handler (l, R_LIB_TYPE_BIN, "bin plugins",
|
|
|
|
&__lib_bin_cb, &__lib_bin_dt, NULL);
|
2010-09-24 21:23:13 +02:00
|
|
|
r_lib_add_handler (l, R_LIB_TYPE_BIN_XTR, "bin xtr plugins",
|
|
|
|
&__lib_bin_xtr_cb, &__lib_bin_xtr_dt, NULL);
|
2010-05-28 17:15:20 +02:00
|
|
|
|
|
|
|
{ /* load plugins everywhere */
|
|
|
|
char *homeplugindir = r_str_home (".radare/plugins");
|
|
|
|
r_lib_opendir (l, getenv ("LIBR_PLUGINS"));
|
|
|
|
r_lib_opendir (l, homeplugindir);
|
|
|
|
r_lib_opendir (l, LIBDIR"/radare2/");
|
|
|
|
}
|
|
|
|
|
2010-10-04 03:46:58 +02:00
|
|
|
while ((c = getopt (argc, argv, "Af:a:B:b:Mm:n:@:VisSzIHelRwO:o:rvLhx")) != -1) {
|
2010-05-28 17:15:20 +02:00
|
|
|
switch(c) {
|
2010-09-24 21:23:13 +02:00
|
|
|
case 'A':
|
|
|
|
action |= ACTION_LISTARCHS;
|
|
|
|
break;
|
|
|
|
case 'a':
|
2010-09-25 03:45:03 +02:00
|
|
|
if (optarg) arch = strdup (optarg);
|
2010-09-24 21:23:13 +02:00
|
|
|
break;
|
2010-10-04 03:46:58 +02:00
|
|
|
case 'f':
|
|
|
|
if (optarg) arch_name = strdup (optarg);
|
|
|
|
break;
|
2010-09-24 21:23:13 +02:00
|
|
|
case 'B':
|
|
|
|
bits = r_num_math (NULL, optarg);
|
|
|
|
break;
|
2010-05-28 17:15:20 +02:00
|
|
|
case 'm':
|
|
|
|
at = r_num_math (NULL, optarg);
|
|
|
|
action |= ACTION_SRCLINE;
|
2010-09-24 21:23:13 +02:00
|
|
|
break;
|
2010-05-28 17:15:20 +02:00
|
|
|
case 'i':
|
|
|
|
action |= ACTION_IMPORTS;
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
action |= ACTION_SYMBOLS;
|
|
|
|
break;
|
|
|
|
case 'S':
|
|
|
|
action |= ACTION_SECTIONS;
|
|
|
|
break;
|
|
|
|
case 'z':
|
|
|
|
action |= ACTION_STRINGS;
|
|
|
|
break;
|
|
|
|
case 'I':
|
|
|
|
action |= ACTION_INFO;
|
|
|
|
break;
|
|
|
|
case 'H':
|
|
|
|
action |= ACTION_FIELDS;
|
|
|
|
break;
|
|
|
|
case 'e':
|
|
|
|
action |= ACTION_ENTRIES;
|
|
|
|
break;
|
2010-05-30 06:06:25 +02:00
|
|
|
case 'M':
|
|
|
|
action |= ACTION_MAIN;
|
|
|
|
break;
|
2010-05-28 17:15:20 +02:00
|
|
|
case 'l':
|
|
|
|
action |= ACTION_LIBS;
|
|
|
|
break;
|
2010-09-10 11:11:38 +02:00
|
|
|
case 'R':
|
|
|
|
action |= ACTION_RELOCS;
|
|
|
|
break;
|
2010-07-29 16:04:18 +02:00
|
|
|
case 'x':
|
|
|
|
action |= ACTION_EXTRACT;
|
|
|
|
break;
|
2010-05-28 17:15:20 +02:00
|
|
|
case 'w':
|
|
|
|
rw = R_TRUE;
|
|
|
|
break;
|
|
|
|
case 'O':
|
|
|
|
op = optarg;
|
|
|
|
action |= ACTION_OPERATION;
|
|
|
|
break;
|
|
|
|
case 'o':
|
|
|
|
output = optarg;
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
rad = R_TRUE;
|
|
|
|
break;
|
|
|
|
case 'v':
|
|
|
|
va = R_TRUE;
|
|
|
|
break;
|
|
|
|
case 'L':
|
|
|
|
r_bin_list (bin);
|
2010-09-24 21:23:13 +02:00
|
|
|
return 1;
|
2010-08-16 14:35:15 +02:00
|
|
|
case 'b':
|
|
|
|
gbaddr = r_num_math (NULL, optarg);
|
|
|
|
break;
|
2010-05-28 17:15:20 +02:00
|
|
|
case '@':
|
|
|
|
at = r_num_math (NULL, optarg);
|
|
|
|
break;
|
2010-08-17 13:15:18 +02:00
|
|
|
case 'n':
|
|
|
|
name = optarg;
|
|
|
|
break;
|
2010-05-28 17:15:20 +02:00
|
|
|
case 'V':
|
2010-07-15 13:34:53 +02:00
|
|
|
printf ("rabin2 v"R2_VERSION"\n");
|
2010-05-28 17:15:20 +02:00
|
|
|
return 0;
|
|
|
|
case 'h':
|
|
|
|
default:
|
|
|
|
action |= ACTION_HELP;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
file = argv[optind];
|
|
|
|
if (action == ACTION_HELP || action == ACTION_UNK || file == NULL)
|
|
|
|
return rabin_show_help ();
|
|
|
|
|
2010-09-24 21:23:13 +02:00
|
|
|
if (!r_bin_load (bin, file, R_FALSE) &&
|
|
|
|
!r_bin_load (bin, file, R_TRUE)) {
|
2010-05-28 17:15:20 +02:00
|
|
|
eprintf ("r_bin: Cannot open '%s'\n", file);
|
2010-09-24 21:23:13 +02:00
|
|
|
return 1;
|
|
|
|
}
|
2010-09-25 03:45:03 +02:00
|
|
|
if (arch) {
|
|
|
|
char *ptr;
|
|
|
|
ptr = strchr (arch, '_');
|
|
|
|
if (ptr) {
|
|
|
|
*ptr = '\0';
|
|
|
|
bits = r_num_math (NULL, ptr+1);
|
|
|
|
}
|
|
|
|
}
|
2010-10-04 03:46:58 +02:00
|
|
|
if (action&ACTION_LISTARCHS ||
|
|
|
|
((arch || bits || arch_name) &&
|
|
|
|
!r_bin_set_arch (bin, arch, bits, arch_name))) {
|
|
|
|
r_bin_list_archs (bin);
|
2010-09-25 03:45:03 +02:00
|
|
|
free (arch);
|
2010-09-24 21:23:13 +02:00
|
|
|
r_bin_free (bin);
|
|
|
|
return 1;
|
2010-05-28 17:15:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (action&ACTION_SECTIONS)
|
|
|
|
rabin_show_sections (at);
|
|
|
|
if (action&ACTION_ENTRIES)
|
|
|
|
rabin_show_entrypoints ();
|
2010-05-30 06:06:25 +02:00
|
|
|
if (action&ACTION_MAIN)
|
|
|
|
rabin_show_main ();
|
2010-05-28 17:15:20 +02:00
|
|
|
if (action&ACTION_IMPORTS)
|
|
|
|
rabin_show_imports (at);
|
|
|
|
if (action&ACTION_SYMBOLS)
|
|
|
|
rabin_show_symbols (at);
|
|
|
|
if (action&ACTION_STRINGS)
|
|
|
|
rabin_show_strings ();
|
|
|
|
if (action&ACTION_INFO)
|
|
|
|
rabin_show_info ();
|
|
|
|
if (action&ACTION_FIELDS)
|
|
|
|
rabin_show_fields();
|
|
|
|
if (action&ACTION_LIBS)
|
|
|
|
rabin_show_libs();
|
2010-09-10 11:11:38 +02:00
|
|
|
if (action&ACTION_RELOCS)
|
|
|
|
rabin_show_relocs();
|
2010-05-28 17:15:20 +02:00
|
|
|
if (action&ACTION_SRCLINE)
|
|
|
|
rabin_show_srcline(at);
|
2010-07-29 16:04:18 +02:00
|
|
|
if (action&ACTION_EXTRACT)
|
2010-10-04 04:01:25 +02:00
|
|
|
rabin_extract ((arch==NULL&&arch_name==NULL&&bits==0));
|
2010-05-28 17:15:20 +02:00
|
|
|
if (op != NULL && action&ACTION_OPERATION)
|
|
|
|
rabin_do_operation (op);
|
|
|
|
|
2010-09-25 03:45:03 +02:00
|
|
|
free (arch);
|
2010-05-28 17:15:20 +02:00
|
|
|
r_bin_free (bin);
|
|
|
|
|
2010-09-24 21:23:13 +02:00
|
|
|
return 0;
|
2010-05-28 17:15:20 +02:00
|
|
|
}
|