mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-08 22:30:30 +00:00
1200 lines
31 KiB
C
1200 lines
31 KiB
C
/* radare - LGPL - Copyright 2009-2016 - pancake */
|
|
#include <stdbool.h>
|
|
#include <string.h>
|
|
|
|
#include "r_crypto.h"
|
|
#include "r_config.h"
|
|
#include "r_cons.h"
|
|
#include "r_core.h"
|
|
#include "r_io.h"
|
|
|
|
R_API int cmd_write_hexpair(RCore* core, const char* pairs) {
|
|
ut8 *buf = malloc (strlen (pairs) + 1);
|
|
int len = r_hex_str2bin (pairs, buf);
|
|
if (len != 0) {
|
|
if (len < 0) {
|
|
len = -len;
|
|
if (len < core->blocksize) {
|
|
buf[len-1] |= core->block[len-1] & 0xf;
|
|
}
|
|
}
|
|
r_core_write_at (core, core->offset, buf, len);
|
|
if (r_config_get_i (core->config, "cfg.wseek")) {
|
|
r_core_seek_delta (core, len);
|
|
}
|
|
r_core_block_read (core, 0);
|
|
} else {
|
|
eprintf ("Error: invalid hexpair string\n");
|
|
}
|
|
free (buf);
|
|
return len;
|
|
}
|
|
|
|
static bool encrypt_or_decrypt_block(RCore *core, const char *algo, const char *key, int direction, const char *iv) {
|
|
//TODO: generalise no_key_mode for all non key encoding/decoding.
|
|
int keylen = key ? strlen (key): 0;
|
|
bool no_key_mode = !strcmp ("base64", algo) || !strcmp ("base91", algo) || !strcmp ("punycode", algo);
|
|
if (no_key_mode || keylen > 0) {
|
|
RCrypto *cry = r_crypto_new ();
|
|
if (r_crypto_use (cry, algo)) {
|
|
ut8 *binkey = malloc (keylen + 1);
|
|
if (binkey) {
|
|
int len = no_key_mode ? 1 : r_hex_str2bin (key, binkey);
|
|
if (len < 1) {
|
|
len = keylen;
|
|
strcpy ((char *)binkey, key);
|
|
} else {
|
|
keylen = len;
|
|
}
|
|
if (r_crypto_set_key (cry, binkey, keylen, 0, direction)) {
|
|
if (iv) {
|
|
ut8 *biniv = malloc (strlen (iv) + 1);
|
|
int ivlen = r_hex_str2bin (iv, biniv);
|
|
if (ivlen < 1) {
|
|
ivlen = strlen(iv);
|
|
strcpy ((char *)biniv, iv);
|
|
}
|
|
if (!r_crypto_set_iv (cry, biniv, ivlen)) {
|
|
eprintf ("Invalid IV.\n");
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
r_crypto_update (cry, (const ut8*)core->block, core->blocksize);
|
|
r_crypto_final (cry, NULL, 0);
|
|
|
|
int result_size = 0;
|
|
ut8 *result = r_crypto_get_output (cry, &result_size);
|
|
if (result) {
|
|
r_io_write_at (core->io, core->offset, result, result_size);
|
|
eprintf ("Written %d bytes\n", result_size);
|
|
free (result);
|
|
}
|
|
} else {
|
|
eprintf ("Invalid key\n");
|
|
}
|
|
free (binkey);
|
|
return 0;
|
|
} else {
|
|
eprintf ("Cannot allocate %d bytes\n", keylen);
|
|
}
|
|
} else {
|
|
eprintf ("Unknown %s algorithm '%s'\n", ((!direction) ? "encryption" : "decryption") ,algo);
|
|
}
|
|
r_crypto_free (cry);
|
|
} else {
|
|
eprintf ("%s key not defined. Use -S [key]\n", ((!direction) ? "Encryption" : "Decryption"));
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static void cmd_write_bits(RCore *core, int set, ut64 val) {
|
|
ut64 ret, orig;
|
|
// used to set/unset bit in current address
|
|
r_core_read_at (core, core->offset, (ut8*)&orig, sizeof (orig));
|
|
if (set) {
|
|
ret = orig | val;
|
|
} else {
|
|
ret = orig & (~(val));
|
|
}
|
|
r_core_write_at (core, core->offset, (const ut8*)&ret, sizeof (ret));
|
|
}
|
|
|
|
static void cmd_write_inc(RCore *core, int size, st64 num) {
|
|
ut64 *v64;
|
|
ut32 *v32;
|
|
ut16 *v16;
|
|
ut8 *v8;
|
|
switch (size) {
|
|
case 1: v8 = (ut8*)core->block; *v8 += num; break;
|
|
case 2: v16 = (ut16*)core->block; *v16 += num; break;
|
|
case 4: v32 = (ut32*)core->block; *v32 += num; break;
|
|
case 8: v64 = (ut64*)core->block; *v64 += num; break;
|
|
}
|
|
// TODO: obey endian here
|
|
r_core_write_at (core, core->offset, core->block, size);
|
|
}
|
|
|
|
static void cmd_write_op (RCore *core, const char *input) {
|
|
ut8 *buf;
|
|
int len;
|
|
const char* help_msg[] = {
|
|
"Usage:","wo[asmdxoArl24]"," [hexpairs] @ addr[!bsize]",
|
|
"wo[aAdlmorwx24]","", "without hexpair values, clipboard is used",
|
|
"woa"," [val]", "+= addition (f.ex: woa 0102)",
|
|
"woA"," [val]","&= and",
|
|
"wod"," [val]", "/= divide",
|
|
"woD","[algo] [key] [IV]","decrypt current block with given algo and key",
|
|
"woe"," [from to] [step] [wsz=1]",".. create sequence",
|
|
"woE"," [algo] [key] [IV]", "encrypt current block with given algo and key",
|
|
"wol"," [val]","<<= shift left",
|
|
"wom"," [val]", "*= multiply",
|
|
"woo"," [val]","|= or",
|
|
"wop[DO]"," [arg]","De Bruijn Patterns",
|
|
"wor"," [val]", ">>= shift right",
|
|
"woR","","random bytes (alias for 'wr $b')",
|
|
"wos"," [val]", "-= substraction",
|
|
"wow"," [val]", "== write looped value (alias for 'wb')",
|
|
"wox"," [val]","^= xor (f.ex: wox 0x90)",
|
|
"wo2"," [val]","2= 2 byte endian swap",
|
|
"wo4"," [val]", "4= 4 byte endian swap",
|
|
NULL
|
|
};
|
|
if (!input[0])
|
|
return;
|
|
switch (input[1]) {
|
|
case 'e':
|
|
if (input[2]!=' ') {
|
|
r_cons_printf ("Usage: 'woe from-to step'\n");
|
|
return;
|
|
}
|
|
/* fallthru */
|
|
case 'a':
|
|
case 's':
|
|
case 'A':
|
|
case 'x':
|
|
case 'r':
|
|
case 'l':
|
|
case 'm':
|
|
case 'd':
|
|
case 'o':
|
|
case 'w':
|
|
case '2':
|
|
case '4':
|
|
if (input[2]) { // parse val from arg
|
|
r_core_write_op (core, input+3, input[1]);
|
|
r_core_block_read (core, 0);
|
|
} else { // use clipboard instead of val
|
|
r_core_write_op (core, NULL, input[1]);
|
|
r_core_block_read (core, 0);
|
|
}
|
|
break;
|
|
case 'R':
|
|
r_core_cmd0 (core, "wr $b");
|
|
break;
|
|
case 'n':
|
|
r_core_write_op (core, "ff", 'x');
|
|
r_core_block_read (core, 0);
|
|
break;
|
|
case 'E': // encrypt
|
|
case 'D': // decrypt
|
|
{
|
|
int direction = (input[1] == 'E') ? 0 : 1;
|
|
const char *algo = NULL;
|
|
const char *key = NULL;
|
|
const char *iv = NULL;
|
|
char *space, *args = strdup (r_str_chop_ro (input+2));
|
|
space = strchr (args, ' ');
|
|
if (space) {
|
|
*space++ = 0;
|
|
key = space;
|
|
space = strchr (key, ' ');
|
|
if (space) {
|
|
*space++ = 0;
|
|
iv = space;
|
|
}
|
|
}
|
|
algo = args;
|
|
if (algo && *algo) {
|
|
encrypt_or_decrypt_block (core, algo, key, direction, iv);
|
|
} else {
|
|
eprintf ("Usage: wo%c [algo] [key] [IV]\n", ((!direction)?'E':'D'));
|
|
eprintf ("Currently supported hashes:\n");
|
|
ut64 bits;
|
|
int i;
|
|
for (i = 0; ; i++) {
|
|
bits = ((ut64)1) << i;
|
|
const char *name = r_hash_name (bits);
|
|
if (!name || !*name) break;
|
|
printf (" %s\n", name);
|
|
}
|
|
eprintf ("Available Encoders/Decoders: \n");
|
|
// TODO: do not hardcode
|
|
eprintf (" base64\n");
|
|
eprintf (" base91\n");
|
|
eprintf (" punycode\n");
|
|
eprintf ("Currently supported crypto algos:\n");
|
|
for (i = 0; ; i++) {
|
|
bits = ((ut64)1) << i;
|
|
const char *name = r_crypto_name (bits);
|
|
if (!name || !*name) break;
|
|
printf (" %s\n", name);
|
|
}
|
|
}
|
|
free (args);
|
|
}
|
|
break;
|
|
case 'p': // debrujin patterns
|
|
switch (input[2]) {
|
|
case 'D': // "wopD"
|
|
len = (int)(input[3]==' ')
|
|
? r_num_math (core->num, input + 3)
|
|
: core->blocksize;
|
|
if (len > 0) {
|
|
/* XXX This seems to fail at generating long patterns (wopD 512K) */
|
|
buf = (ut8*)r_debruijn_pattern (len, 0, NULL); //debruijn_charset);
|
|
if (buf) {
|
|
const ut8 *ptr = buf;
|
|
ut64 addr = core->offset;
|
|
while (true) {
|
|
int res = r_core_write_at (core, addr, ptr, len);
|
|
if (res < 1 || len == res) {
|
|
break;
|
|
}
|
|
if (res < len) {
|
|
ptr += res;
|
|
len -= res;
|
|
addr += res;
|
|
}
|
|
}
|
|
free (buf);
|
|
} else {
|
|
eprintf ("Couldn't generate pattern of length %d\n", len);
|
|
}
|
|
}
|
|
break;
|
|
case 'O': // "wopO"
|
|
len = (int)(input[3]==' ')
|
|
? r_num_math (core->num, input + 3)
|
|
: core->blocksize;
|
|
core->num->value = r_debruijn_offset (len, r_config_get_i (core->config, "cfg.bigendian"));
|
|
r_cons_printf ("%"PFMT64d"\n", core->num->value);
|
|
break;
|
|
case '\0':
|
|
case '?':
|
|
default:
|
|
{
|
|
const char* wop_help_msg[] = {
|
|
"Usage:","wop[DO]"," len @ addr | value",
|
|
"wopD"," len [@ addr]","Write a De Bruijn Pattern of length 'len' at address 'addr'",
|
|
"wopO"," value", "Finds the given value into a De Bruijn Pattern at current offset",
|
|
NULL
|
|
};
|
|
r_core_cmd_help (core, wop_help_msg);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case '\0':
|
|
case '?':
|
|
default:
|
|
r_core_cmd_help (core, help_msg);
|
|
break;
|
|
}
|
|
}
|
|
|
|
#define WSEEK(x,y) if (wseek)r_core_seek_delta (x,y)
|
|
static void cmd_write_value (RCore *core, const char *input) {
|
|
int type = 0;
|
|
ut64 off = 0LL;
|
|
ut8 buf[sizeof(ut64)];
|
|
int wseek = r_config_get_i (core->config, "cfg.wseek");
|
|
|
|
if (!input)
|
|
return;
|
|
|
|
if (input[0])
|
|
switch (input[1]) {
|
|
case '?':
|
|
{
|
|
const char* help_msg[] = {
|
|
"Usage:", "wv[size] [value]", "write value of given size",
|
|
"wv1", " 234", "write one byte with this value",
|
|
"wv", " 0x834002", "write dword with this value",
|
|
"Supported sizes are:", "1, 2, 4, 8", "",
|
|
NULL};
|
|
r_core_cmd_help (core, help_msg);
|
|
return;
|
|
}
|
|
case '1': type = 1; break;
|
|
case '2': type = 2; break;
|
|
case '4': type = 4; break;
|
|
case '8': type = 8; break;
|
|
}
|
|
if (input && input[0] && input[1] && input[2]) {
|
|
off = r_num_math (core->num, input+2);
|
|
}
|
|
if (core->file) {
|
|
r_io_use_desc (core->io, core->file->desc);
|
|
}
|
|
ut64 res = r_io_seek (core->io, core->offset, R_IO_SEEK_SET);
|
|
if (res == UT64_MAX) return;
|
|
if (type == 0)
|
|
type = (off&UT64_32U)? 8: 4;
|
|
switch (type) {
|
|
case 1:
|
|
r_write_ble8 (buf, (ut8)(off & UT8_MAX));
|
|
r_io_write (core->io, buf, 1);
|
|
WSEEK (core, 1);
|
|
break;
|
|
case 2:
|
|
r_write_le16 (buf, (ut16)(off & UT16_MAX));
|
|
r_io_write (core->io, buf, 2);
|
|
WSEEK (core, 2);
|
|
break;
|
|
case 4:
|
|
r_write_le32 (buf, (ut32)(off & UT32_MAX));
|
|
r_io_write (core->io, buf, 4);
|
|
WSEEK (core, 4);
|
|
break;
|
|
case 8:
|
|
r_write_le64 (buf, off);
|
|
r_io_write (core->io, buf, 8);
|
|
WSEEK (core, 8);
|
|
break;
|
|
}
|
|
r_core_block_read (core, 0);
|
|
}
|
|
|
|
static bool cmd_wf(RCore *core, const char *input) {
|
|
ut8 *buf;
|
|
int size;
|
|
const char *arg = input + ((input[1] == ' ') ? 2 : 1);
|
|
int wseek = r_config_get_i (core->config, "cfg.wseek");
|
|
char *p, *a = r_str_chop (strdup (arg));
|
|
// XXX: file names cannot contain spaces
|
|
p = strchr (a, ' ');
|
|
if (p) *p++ = 0;
|
|
|
|
if (*arg =='?' || !*arg) {
|
|
eprintf ("Usage: wf [file] ([size] ([offset]))\n");
|
|
}
|
|
if (!strcmp (arg, "-")) {
|
|
char *out = r_core_editor (core, NULL, NULL);
|
|
if (out) {
|
|
r_io_write_at (core->io, core->offset,
|
|
(ut8*)out, strlen (out));
|
|
free (out);
|
|
}
|
|
}
|
|
if ((buf = (ut8*) r_file_slurp (a, &size))) {
|
|
int u_size = size;
|
|
int u_offset = 0;
|
|
u_size = r_num_math (core->num, p);
|
|
if (u_size < 1) u_size = size;
|
|
if (p) {
|
|
*p++ = 0;
|
|
u_offset = r_num_math (core->num, p);
|
|
if (u_offset > size) {
|
|
eprintf ("Invalid offset\n");
|
|
free (buf);
|
|
return false;
|
|
}
|
|
}
|
|
r_io_use_desc (core->io, core->file->desc);
|
|
r_io_write_at (core->io, core->offset, buf + u_offset, u_size);
|
|
WSEEK (core, size);
|
|
free (buf);
|
|
r_core_block_read (core, 0);
|
|
} else {
|
|
eprintf ("Cannot open file '%s'\n", arg);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/* TODO: simplify using r_write */
|
|
static int cmd_write(void *data, const char *input) {
|
|
int wseek, i, size, len = strlen (input);
|
|
RCore *core = (RCore *)data;
|
|
char *tmp, *str, *ostr;
|
|
const char *arg, *filename;
|
|
char _fn[32];
|
|
ut64 off;
|
|
ut8 *buf;
|
|
st64 num = 0;
|
|
const char* help_msg[] = {
|
|
"Usage:","w[x] [str] [<file] [<<EOF] [@addr]","",
|
|
"w","[1248][+-][n]","increment/decrement byte,word..",
|
|
"w"," foobar","write string 'foobar'",
|
|
"w0"," [len]","write 'len' bytes with value 0x00",
|
|
"w6","[de] base64/hex","write base64 [d]ecoded or [e]ncoded string",
|
|
"wa"," push ebp","write opcode, separated by ';' (use '\"' around the command)",
|
|
"waf"," file","assemble file and write bytes",
|
|
"wao"," op","modify opcode (change conditional of jump. nop, etc)",
|
|
"wA"," r 0","alter/modify opcode at current seek (see wA?)",
|
|
"wb"," 010203","fill current block with cyclic hexpairs",
|
|
"wB","[-]0xVALUE","set or unset bits with given value",
|
|
"wc","","list all write changes",
|
|
"wc","[ir*?]","write cache undo/commit/reset/list (io.cache)",
|
|
"wd"," [off] [n]","duplicate N bytes from offset at current seek (memcpy) (see y?)",
|
|
"we","[nNsxX] [arg]","extend write operations (insert instead of replace)",
|
|
"wf"," -|file","write contents of file at current offset",
|
|
"wh"," r2","whereis/which shell command",
|
|
"wm"," f0ff","set binary mask hexpair to be used as cyclic write mask",
|
|
"wo?"," hex","write in block with operation. 'wo?' fmi",
|
|
"wp"," -|file","apply radare patch file. See wp? fmi",
|
|
"wr"," 10","write 10 random bytes",
|
|
"ws"," pstring","write 1 byte for length and then the string",
|
|
"wt"," file [sz]","write to file (from current seek, blocksize or sz bytes)",
|
|
"ww"," foobar","write wide string 'f\\x00o\\x00o\\x00b\\x00a\\x00r\\x00'",
|
|
"wx[fs]"," 9090","write two intel nops (from wxfile or wxseek)",
|
|
"wv"," eip+34","write 32-64 bit value",
|
|
"wz"," string","write zero terminated string (like w + \\x00)",
|
|
NULL
|
|
};
|
|
|
|
if (!input)
|
|
return 0;
|
|
|
|
wseek = r_config_get_i (core->config, "cfg.wseek");
|
|
str = ostr = strdup ((input&&*input)?input+1:"");
|
|
_fn[0] = 0;
|
|
|
|
switch (*input) {
|
|
case 'B':
|
|
switch (input[1]) {
|
|
case ' ':
|
|
cmd_write_bits (core, 1, r_num_math (core->num, input+2));
|
|
break;
|
|
case '-':
|
|
cmd_write_bits (core, 0, r_num_math (core->num, input+2));
|
|
break;
|
|
default:
|
|
eprintf ("Usage: wB 0x2000 # or wB-0x2000\n");
|
|
break;
|
|
}
|
|
break;
|
|
case '0':
|
|
{
|
|
ut64 len = r_num_math (core->num, input+1);
|
|
if (len>0) {
|
|
ut8 *buf = calloc (1, len);
|
|
if (buf) {
|
|
r_io_write (core->io, buf, len);
|
|
free (buf);
|
|
} else eprintf ("Cannot allocate %d bytes\n", (int)len);
|
|
}
|
|
}
|
|
break;
|
|
case '1':
|
|
case '2':
|
|
case '4':
|
|
case '8':
|
|
if (input[1] && input[2]) {
|
|
if (input[1]==input[2]) {
|
|
num = 1;
|
|
} else num = r_num_math (core->num, input+2);
|
|
}
|
|
switch (input[2] ? input[1] : 0) {
|
|
case '+':
|
|
cmd_write_inc (core, *input-'0', num);
|
|
break;
|
|
case '-':
|
|
cmd_write_inc (core, *input-'0', -num);
|
|
break;
|
|
default:
|
|
eprintf ("Usage: w[1248][+-][num] # inc/dec byte/word/..\n");
|
|
}
|
|
break;
|
|
case '6':
|
|
{
|
|
int fail = 0;
|
|
ut8 *buf;
|
|
int len, str_len;
|
|
const char *str;
|
|
|
|
if (input[1] && input[2] != ' ')
|
|
fail = 1;
|
|
|
|
if (input[1] && input[2] && input[3])
|
|
str = input + 3;
|
|
else
|
|
str = "";
|
|
str_len = strlen (str) + 1;
|
|
if (!fail) {
|
|
switch (input[1]) {
|
|
case 'd':
|
|
buf = malloc (str_len);
|
|
len = r_base64_decode (buf, str, 0);
|
|
if(len == 0) {
|
|
free(buf);
|
|
fail = 1;
|
|
}
|
|
break;
|
|
case 'e':
|
|
{
|
|
ut8 *bin_buf = malloc (str_len);
|
|
const int bin_len = r_hex_str2bin (str, bin_buf);
|
|
if (bin_len <= 0) {
|
|
fail = 1;
|
|
} else {
|
|
buf = calloc (str_len + 1, 4);
|
|
len = r_base64_encode ((char *)buf, bin_buf, bin_len);
|
|
if(len == 0) {
|
|
free (buf);
|
|
fail = 1;
|
|
}
|
|
}
|
|
free (bin_buf);
|
|
break;
|
|
}
|
|
default:
|
|
fail = 1;
|
|
break;
|
|
}
|
|
}
|
|
if (!fail) {
|
|
r_core_write_at (core, core->offset, buf, len);
|
|
WSEEK (core, len);
|
|
r_core_block_read (core, 0);
|
|
free (buf);
|
|
} else {
|
|
eprintf ("Usage: w6[de] base64/hex\n");
|
|
}
|
|
break;
|
|
}
|
|
case 'h':
|
|
{
|
|
char *p = strchr (input, ' ');
|
|
if (p) {
|
|
while (*p==' ') p++;
|
|
p = r_file_path (p);
|
|
if (p) {
|
|
r_cons_println (p);
|
|
free (p);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case 'e':
|
|
{
|
|
ut64 addr = 0, len = 0, b_size = 0;
|
|
st64 dist = 0;
|
|
ut8* bytes = NULL;
|
|
int cmd_suc = false;
|
|
char *input_shadow = NULL, *p = NULL;
|
|
|
|
switch (input[1]) {
|
|
case 'n':
|
|
if (input[2] == ' ') {
|
|
len = *input ? r_num_math (core->num, input+3) : 0;
|
|
if (len > 0) {
|
|
const ut64 cur_off = core->offset;
|
|
cmd_suc = r_core_extend_at (core, core->offset, len);
|
|
core->offset = cur_off;
|
|
r_core_block_read (core, 0);
|
|
}
|
|
}
|
|
break;
|
|
case 'N':
|
|
if (input[2] == ' ') {
|
|
input += 3;
|
|
while (*input && *input == ' ') input++;
|
|
addr = r_num_math (core->num, input);
|
|
while (*input && *input != ' ') input++;
|
|
input++;
|
|
len = *input ? r_num_math (core->num, input) : 0;
|
|
if (len > 0){
|
|
ut64 cur_off = core->offset;
|
|
cmd_suc = r_core_extend_at (core, addr, len);
|
|
cmd_suc = r_core_seek (core, cur_off, 1);
|
|
core->offset = addr;
|
|
r_core_block_read (core, 0);
|
|
}
|
|
}
|
|
break;
|
|
case 'x':
|
|
if (input[2] == ' ') {
|
|
input += 2;
|
|
len = *input ? strlen (input) : 0;
|
|
bytes = len > 1? malloc (len+1) : NULL;
|
|
len = bytes ? r_hex_str2bin (input, bytes) : 0;
|
|
if (len > 0) {
|
|
ut64 cur_off = core->offset;
|
|
cmd_suc = r_core_extend_at (core, cur_off, len);
|
|
if (cmd_suc) {
|
|
r_core_write_at (core, cur_off, bytes, len);
|
|
}
|
|
core->offset = cur_off;
|
|
r_core_block_read (core, 0);
|
|
}
|
|
free (bytes);
|
|
}
|
|
break;
|
|
case 'X':
|
|
if (input[2] == ' ') {
|
|
addr = r_num_math (core->num, input+3);
|
|
input += 3;
|
|
while (*input && *input != ' ') input++;
|
|
input++;
|
|
len = *input ? strlen (input) : 0;
|
|
bytes = len > 1? malloc (len+1) : NULL;
|
|
len = bytes ? r_hex_str2bin (input, bytes) : 0;
|
|
if (len > 0) {
|
|
//ut64 cur_off = core->offset;
|
|
cmd_suc = r_core_extend_at (core, addr, len);
|
|
if (cmd_suc) {
|
|
r_core_write_at (core, addr, bytes, len);
|
|
}
|
|
core->offset = addr;
|
|
r_core_block_read (core, 0);
|
|
}
|
|
free (bytes);
|
|
}
|
|
break;
|
|
case 's':
|
|
input += 3;
|
|
while (*input && *input == ' ') input++;
|
|
len = strlen (input);
|
|
input_shadow = len > 0? malloc (len+1): 0;
|
|
|
|
// since the distance can be negative,
|
|
// the r_num_math will perform an unwanted operation
|
|
// the solution is to tokenize the string :/
|
|
if (input_shadow) {
|
|
strncpy (input_shadow, input, len+1);
|
|
p = strtok (input_shadow, " ");
|
|
addr = p && *p ? r_num_math (core->num, p) : 0;
|
|
|
|
p = strtok (NULL, " ");
|
|
dist = p && *p ? r_num_math (core->num, p) : 0;
|
|
|
|
p = strtok (NULL, " ");
|
|
b_size = p && *p ? r_num_math (core->num, p) : 0;
|
|
if (dist != 0){
|
|
r_core_shift_block (core, addr, b_size, dist);
|
|
r_core_seek (core, addr, 1);
|
|
cmd_suc = true;
|
|
}
|
|
}
|
|
free (input_shadow);
|
|
break;
|
|
case '?':
|
|
default:
|
|
cmd_suc = false;
|
|
}
|
|
|
|
|
|
if (cmd_suc == false) {
|
|
const char* help_msg[] = {
|
|
"Usage", "", "write extend",
|
|
"wen", " <num>", "insert num null bytes at current offset",
|
|
"wex", " <hex_bytes>", "insert bytes at current offset",
|
|
"weN", " <addr> <len>", "insert bytes at address",
|
|
"weX", " <addr> <hex_bytes>", "insert bytes at address",
|
|
"wes", " <addr> <dist> <block_size>", "shift a blocksize left or write in the editor",
|
|
NULL};
|
|
r_core_cmd_help (core, help_msg);
|
|
}
|
|
}
|
|
break;
|
|
case 'p':
|
|
if (input[1]=='-' || (input[1]==' ' && input[2]=='-')) {
|
|
char *out = r_core_editor (core, NULL, NULL);
|
|
if (out) {
|
|
r_core_patch (core, out);
|
|
free (out);
|
|
}
|
|
} else {
|
|
if (input[1]==' ' && input[2]) {
|
|
char *data = r_file_slurp (input+2, NULL);
|
|
if (data) {
|
|
r_core_patch (core, data);
|
|
free (data);
|
|
}
|
|
} else {
|
|
eprintf ("Usage: wp [-|r2patch-file]\n"
|
|
"TODO: rapatch format documentation here\n");
|
|
}
|
|
}
|
|
break;
|
|
case 'u':
|
|
// TODO: implement it in an API RCore.write_unified_hexpatch() is ETOOLONG
|
|
if (input[1]==' ') {
|
|
char *data = r_file_slurp (input+2, NULL);
|
|
if (data) {
|
|
char sign = ' ';
|
|
int line = 0, offs = 0, hexa = 0;
|
|
int newline = 1;
|
|
for (i=0; data[i]; i++) {
|
|
switch (data[i]) {
|
|
case '+':
|
|
if (newline)
|
|
sign = 1;
|
|
break;
|
|
case '-':
|
|
if (newline) {
|
|
sign = 0;
|
|
offs = i + ((data[i+1]==' ')?2:1);
|
|
}
|
|
break;
|
|
case ' ':
|
|
data[i] = 0;
|
|
if (sign) {
|
|
if (!line) line = i+1;
|
|
else
|
|
if (!hexa) hexa = i+1;
|
|
}
|
|
break;
|
|
case '\r':
|
|
break;
|
|
case '\n':
|
|
newline = 1;
|
|
if (sign == -1) {
|
|
offs = 0;
|
|
line = 0;
|
|
hexa = 0;
|
|
} else if (sign) {
|
|
if (offs && hexa) {
|
|
r_cons_printf ("wx %s @ %s\n", data+hexa, data+offs);
|
|
} else eprintf ("food\n");
|
|
offs = 0;
|
|
line = 0;
|
|
} else hexa = 0;
|
|
sign = -1;
|
|
continue;
|
|
}
|
|
newline = 0;
|
|
}
|
|
free (data);
|
|
}
|
|
} else {
|
|
eprintf ("|Usage: wu [unified-diff-patch] # see 'cu'\n");
|
|
}
|
|
break;
|
|
case 'r': //wr
|
|
off = r_num_math (core->num, input+1);
|
|
len = (int)off;
|
|
if (len > 0) {
|
|
buf = malloc (len);
|
|
if (buf != NULL) {
|
|
r_num_irand ();
|
|
for (i=0; i<len; i++)
|
|
buf[i] = r_num_rand (256);
|
|
r_core_write_at (core, core->offset, buf, len);
|
|
WSEEK (core, len);
|
|
free (buf);
|
|
} else eprintf ("Cannot allocate %d bytes\n", len);
|
|
}
|
|
break;
|
|
case 'A':
|
|
switch (input[1]) {
|
|
case ' ':
|
|
if (input[2] && input[3]==' ') {
|
|
r_asm_set_pc (core->assembler, core->offset);
|
|
eprintf ("modify (%c)=%s\n", input[2], input+4);
|
|
len = r_asm_modify (core->assembler, core->block, input[2],
|
|
r_num_math (core->num, input+4));
|
|
eprintf ("len=%d\n", len);
|
|
if (len>0) {
|
|
r_core_write_at (core, core->offset, core->block, len);
|
|
WSEEK (core, len);
|
|
} else eprintf ("r_asm_modify = %d\n", len);
|
|
} else eprintf ("Usage: wA [type] [value]\n");
|
|
break;
|
|
case '?':
|
|
default:
|
|
{
|
|
const char* help_msg[] = {
|
|
"Usage:", " wA", "[type] [value]",
|
|
"Types", "", "",
|
|
"r", "", "raw write value",
|
|
"v", "", "set value (taking care of current address)",
|
|
"d", "", "destination register",
|
|
"0", "", "1st src register",
|
|
"1", "", "2nd src register",
|
|
"Example:", "wA r 0", "# e800000000",
|
|
NULL};
|
|
r_core_cmd_help (core, help_msg);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case 'c':
|
|
switch (input[1]) {
|
|
case 'i':
|
|
r_io_cache_commit (core->io, 0, UT64_MAX);
|
|
r_core_block_read (core, 0);
|
|
break;
|
|
case 'r':
|
|
r_io_cache_reset (core->io, true);
|
|
/* Before loading the core block we have to make sure that if
|
|
* the cache wrote past the original EOF these changes are no
|
|
* longer displayed. */
|
|
memset (core->block, 0xff, core->blocksize);
|
|
r_core_block_read (core, 0);
|
|
break;
|
|
case '+':
|
|
if (input[2]=='*') {
|
|
//r_io_cache_reset (core->io, true);
|
|
eprintf ("TODO\n");
|
|
} else if (input[2]==' ') {
|
|
char *p = strchr (input+3, ' ');
|
|
ut64 to, from;
|
|
from = r_num_math (core->num, input+3);
|
|
if (p) {
|
|
*p = 0;
|
|
to = r_num_math (core->num, input+3);
|
|
if (to<from) {
|
|
eprintf ("Invalid range (from>to)\n");
|
|
return 0;
|
|
}
|
|
} else {
|
|
to = from + core->blocksize;
|
|
}
|
|
r_io_cache_commit (core->io, from, to);
|
|
} else {
|
|
eprintf ("Invalidate write cache at 0x%08"PFMT64x"\n", core->offset);
|
|
r_io_cache_commit (core->io, core->offset, core->offset+1);
|
|
}
|
|
break;
|
|
case '-':
|
|
if (input[2]=='*') {
|
|
r_io_cache_reset (core->io, true);
|
|
} else if (input[2]==' ') {
|
|
char *p = strchr (input+3, ' ');
|
|
ut64 to, from;
|
|
if (p) {
|
|
*p = 0;
|
|
from = r_num_math (core->num, input+3);
|
|
to = r_num_math (core->num, input+3);
|
|
if (to<from) {
|
|
eprintf ("Invalid range (from>to)\n");
|
|
return 0;
|
|
}
|
|
} else {
|
|
from = r_num_math (core->num, input+3);
|
|
to = from + core->blocksize;
|
|
}
|
|
r_io_cache_invalidate (core->io, from, to);
|
|
} else {
|
|
eprintf ("Invalidate write cache at 0x%08"PFMT64x"\n", core->offset);
|
|
r_io_cache_invalidate (core->io, core->offset, core->offset+core->blocksize);
|
|
}
|
|
/* See 'r' above. */
|
|
memset (core->block, 0xff, core->blocksize);
|
|
r_core_block_read (core, 0);
|
|
break;
|
|
case '?':
|
|
{
|
|
const char* help_msg[] = {
|
|
"Usage:", "wc[ir+-*?]"," # NOTE: Uses io.cache=true",
|
|
"wc","","list all write changes",
|
|
"wc-"," [from] [to]","remove write op at curseek or given addr",
|
|
"wc+"," [addr]","commit change from cache to io",
|
|
"wc*","","\"\" in radare commands",
|
|
"wcr","","reset all write changes in cache",
|
|
"wci","","commit write cache",
|
|
NULL
|
|
};
|
|
r_core_cmd_help (core, help_msg);
|
|
}
|
|
break;
|
|
case '*':
|
|
r_io_cache_list (core->io, 1);
|
|
break;
|
|
case 'j':
|
|
r_io_cache_list (core->io, 2);
|
|
break;
|
|
case '\0':
|
|
//if (!r_config_get_i (core->config, "io.cache"))
|
|
// eprintf ("[warning] e io.cache must be true\n");
|
|
r_io_cache_list (core->io, 0);
|
|
break;
|
|
}
|
|
break;
|
|
case ' ': // "w"
|
|
/* write string */
|
|
len = r_str_unescape (str);
|
|
r_core_write_at (core, core->offset, (const ut8*)str, len);
|
|
#if 0
|
|
r_io_use_desc (core->io, core->file->desc);
|
|
r_io_write_at (core->io, core->offset, (const ut8*)str, len);
|
|
#endif
|
|
WSEEK (core, len);
|
|
r_core_block_read (core, 0);
|
|
break;
|
|
case 'z': // "wz"
|
|
/* write zero-terminated string */
|
|
len = r_str_unescape (str);
|
|
r_core_write_at (core, core->offset, (const ut8*)str + 1, len);
|
|
if (len > 0) {
|
|
core->num->value = len;
|
|
} else {
|
|
core->num->value = 0;
|
|
}
|
|
#if 0
|
|
r_io_use_desc (core->io, core->file->desc);
|
|
#endif
|
|
WSEEK (core, len + 1);
|
|
r_core_block_read (core, 0);
|
|
break;
|
|
case 't': // "wt"
|
|
if (*str == '?' || *str == '\0') {
|
|
eprintf ("Usage: wt[a] file [size] write 'size' bytes in current block to file\n");
|
|
free (ostr);
|
|
return 0;
|
|
} else {
|
|
int append = 0;
|
|
st64 sz = core->blocksize;
|
|
if (*str=='a') { // "wta"
|
|
append = 1;
|
|
str++;
|
|
if (str[0]==' ') {
|
|
filename = str+1;
|
|
} else {
|
|
const char* prefix = r_config_get (core->config, "cfg.prefixdump");
|
|
snprintf (_fn, sizeof (_fn), "%s.0x%08"PFMT64x, prefix, core->offset);
|
|
filename = _fn;
|
|
}
|
|
} else if (*str != ' ') {
|
|
const char* prefix = r_config_get (core->config, "cfg.prefixdump");
|
|
snprintf (_fn, sizeof(_fn), "%s.0x%08"PFMT64x, prefix, core->offset);
|
|
filename = _fn;
|
|
} else filename = str+1;
|
|
tmp = strchr (str+1, ' ');
|
|
if (tmp) {
|
|
sz = (st64) r_num_math (core->num, tmp+1);
|
|
if (!sz) {
|
|
sz = core->blocksize;
|
|
}
|
|
*tmp = 0;
|
|
if (sz<1) eprintf ("Invalid length\n");
|
|
else r_core_dump (core, filename, core->offset, (ut64)sz, append);
|
|
} else {
|
|
if (!r_file_dump (filename, core->block, core->blocksize, append)) {
|
|
sz = 0;
|
|
} else sz = core->blocksize;
|
|
}
|
|
eprintf ("Dumped %"PFMT64d" bytes from 0x%08"PFMT64x" into %s\n",
|
|
sz, core->offset, filename);
|
|
}
|
|
break;
|
|
case 'f':
|
|
cmd_wf (core, input);
|
|
break;
|
|
case 'w':
|
|
str++;
|
|
len = (len - 1) << 1;
|
|
tmp = (len > 0) ? malloc (len + 1) : NULL;
|
|
if (tmp) {
|
|
for (i=0; i<len; i++) {
|
|
if (i%2) tmp[i] = 0;
|
|
else tmp[i] = str[i>>1];
|
|
}
|
|
str = tmp;
|
|
r_io_use_desc (core->io, core->file->desc);
|
|
r_io_write_at (core->io, core->offset, (const ut8*)str, len);
|
|
WSEEK (core, len);
|
|
r_core_block_read (core, 0);
|
|
free (tmp);
|
|
} else {
|
|
eprintf ("Cannot malloc %d\n", len);
|
|
}
|
|
break;
|
|
case 'x':
|
|
switch (input[1]) {
|
|
case 'f': // "wxf"
|
|
arg = (const char *)(input + ((input[2]==' ')? 3: 2));
|
|
if (!strcmp (arg, "-")) {
|
|
int len;
|
|
ut8 *out;
|
|
char *in = r_core_editor (core, NULL, NULL);
|
|
if (in) {
|
|
out = (ut8 *)strdup (in);
|
|
if (out) {
|
|
len = r_hex_str2bin (in, out);
|
|
if (len > 0) {
|
|
r_io_write_at (core->io, core->offset, out, len);
|
|
core->num->value = len;
|
|
} else {
|
|
core->num->value = 0;
|
|
}
|
|
free (out);
|
|
}
|
|
free (in);
|
|
}
|
|
} else if ((buf = r_file_slurp_hexpairs (arg, &size))) {
|
|
r_io_use_desc (core->io, core->file->desc);
|
|
if (r_io_write_at (core->io, core->offset, buf, size) > 0) {
|
|
core->num->value = size;
|
|
WSEEK (core, size);
|
|
}
|
|
free (buf);
|
|
r_core_block_read (core, 0);
|
|
} else eprintf ("Cannot open file '%s'\n", arg);
|
|
break;
|
|
case 's': // "wxs"
|
|
{
|
|
int len = cmd_write_hexpair (core, input + 1);
|
|
if (len > 0) {
|
|
r_core_seek_delta (core, len);
|
|
core->num->value = len;
|
|
} else {
|
|
core->num->value = 0;
|
|
}
|
|
}
|
|
break;
|
|
case ' ': // "wx ..."
|
|
cmd_write_hexpair (core, input + 1);
|
|
break;
|
|
default:
|
|
{
|
|
const char* help_msg[] = {
|
|
"Usage:", "wx[f] [arg]", "",
|
|
"wx", " 9090", "write two intel nops",
|
|
"wxf", " -|file", "write contents of hexpairs file here",
|
|
"wxs", " 9090", "write hexpairs and seek at the end",
|
|
NULL};
|
|
r_core_cmd_help (core, help_msg);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case 'a':
|
|
switch (input[1]) {
|
|
case 'o': // "wao"
|
|
if (input[2] == ' ')
|
|
r_core_hack (core, input+3);
|
|
else r_core_hack_help (core);
|
|
break;
|
|
case ' ':
|
|
case '*':
|
|
{ const char *file = input[1]=='*'? input+2: input+1;
|
|
RAsmCode *acode;
|
|
r_asm_set_pc (core->assembler, core->offset);
|
|
acode = r_asm_massemble (core->assembler, file);
|
|
if (acode) {
|
|
if (input[1]=='*') {
|
|
cmd_write_hexpair(core, acode->buf_hex);
|
|
} else {
|
|
if (r_config_get_i (core->config, "scr.prompt"))
|
|
eprintf ("Written %d bytes (%s) = wx %s\n", acode->len, input+2, acode->buf_hex);
|
|
r_core_write_at (core, core->offset, acode->buf, acode->len);
|
|
WSEEK (core, acode->len);
|
|
r_core_block_read (core, 0);
|
|
}
|
|
r_asm_code_free (acode);
|
|
}
|
|
} break;
|
|
case 'f': // "wof"
|
|
if ((input[2]==' '||input[2]=='*')) {
|
|
const char *file = input[2]=='*'? input+4: input+3;
|
|
RAsmCode *acode;
|
|
r_asm_set_pc (core->assembler, core->offset);
|
|
acode = r_asm_assemble_file (core->assembler, file);
|
|
if (acode) {
|
|
if (input[2]=='*') {
|
|
cmd_write_hexpair(core, acode->buf_hex);
|
|
} else {
|
|
if (r_config_get_i (core->config, "scr.prompt"))
|
|
eprintf ("Written %d bytes (%s)=wx %s\n", acode->len, input+1, acode->buf_hex);
|
|
r_core_write_at (core, core->offset, acode->buf, acode->len);
|
|
WSEEK (core, acode->len);
|
|
r_core_block_read (core, 0);
|
|
}
|
|
r_asm_code_free (acode);
|
|
} else eprintf ("Cannot assemble file\n");
|
|
} else eprintf ("Wrong argument\n");
|
|
break;
|
|
default:
|
|
{
|
|
const char* help_msg[] = {
|
|
"Usage:", "wa[of*] [arg]", "",
|
|
"wa", " nop", "write nopcode using asm.arch and asm.bits",
|
|
"wa*", " mov eax, 33", "show 'wx' op with hexpair bytes of assembled opcode",
|
|
"\"wa nop;nop\"", "" , "assemble more than one instruction (note the quotes)",
|
|
"waf", "foo.asm" , "assemble file and write bytes",
|
|
"wao?", "", "show help for assembler operation on current opcode (hack)",
|
|
NULL};
|
|
r_core_cmd_help (core, help_msg);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case 'b': // "wb"
|
|
{
|
|
int len = strlen (input);
|
|
ut8 *buf = malloc (len+1);
|
|
if (buf) {
|
|
len = r_hex_str2bin (input+1, buf);
|
|
if (len > 0) {
|
|
r_mem_copyloop (core->block, buf, core->blocksize, len);
|
|
r_core_write_at (core, core->offset, core->block, core->blocksize);
|
|
WSEEK (core, core->blocksize);
|
|
r_core_block_read (core, 0);
|
|
} else eprintf ("Wrong argument\n");
|
|
free (buf);
|
|
} else eprintf ("Cannot malloc %d\n", len+1);
|
|
}
|
|
break;
|
|
case 'm':
|
|
size = r_hex_str2bin (input+1, (ut8*)str);
|
|
switch (input[1]) {
|
|
case '\0':
|
|
eprintf ("Current write mask: TODO\n");
|
|
// TODO
|
|
break;
|
|
case '?':
|
|
break;
|
|
case '-':
|
|
r_io_set_write_mask (core->io, 0, 0);
|
|
eprintf ("Write mask disabled\n");
|
|
break;
|
|
case ' ':
|
|
if (size>0) {
|
|
r_io_use_desc (core->io, core->file->desc);
|
|
r_io_set_write_mask (core->io, (const ut8*)str, size);
|
|
WSEEK (core, size);
|
|
eprintf ("Write mask set to '");
|
|
for (i=0; i<size; i++)
|
|
eprintf ("%02x", str[i]);
|
|
eprintf ("'\n");
|
|
} else eprintf ("Invalid string\n");
|
|
break;
|
|
}
|
|
break;
|
|
case 'v':
|
|
cmd_write_value (core, input);
|
|
break;
|
|
case 'o':
|
|
cmd_write_op (core, input);
|
|
break;
|
|
case 'd':
|
|
if (input[1] && input[1]==' ') {
|
|
char *arg, *inp = strdup (input+2);
|
|
arg = strchr (inp, ' ');
|
|
if (arg) {
|
|
*arg = 0;
|
|
ut64 addr = r_num_math (core->num, input+2);
|
|
ut64 len = r_num_math (core->num, arg+1);
|
|
ut8 *data = malloc (len);
|
|
r_io_read_at (core->io, addr, data, len);
|
|
r_io_write_at (core->io, core->offset, data, len);
|
|
free (data);
|
|
} else eprintf ("See wd?\n");
|
|
free (inp);
|
|
} else eprintf ("Usage: wd [source-offset] [length] @ [dest-offset]\n");
|
|
break;
|
|
case 's':
|
|
if (str && *str && str[1]) {
|
|
len = r_str_unescape (str+1);
|
|
if (len>255) {
|
|
eprintf ("Too large\n");
|
|
} else {
|
|
ut8 ulen = (ut8)len;
|
|
r_core_write_at (core, core->offset, &ulen, 1);
|
|
r_core_write_at (core, core->offset+1, (const ut8*)str+1, len);
|
|
WSEEK (core, len);
|
|
r_core_block_read (core, 0);
|
|
}
|
|
} else eprintf ("Too short.\n");
|
|
break;
|
|
default:
|
|
case '?':
|
|
if (core->oobi) {
|
|
eprintf ("Writing oobi buffer!\n");
|
|
r_io_use_desc (core->io, core->file->desc);
|
|
r_io_write (core->io, core->oobi, core->oobi_len);
|
|
WSEEK (core, core->oobi_len);
|
|
r_core_block_read (core, 0);
|
|
} else {
|
|
r_core_cmd_help (core, help_msg);
|
|
}
|
|
break;
|
|
}
|
|
R_FREE (ostr);
|
|
return 0;
|
|
}
|
|
|