2016-10-26 23:16:31 +02:00
|
|
|
/* radare - LGPL - Copyright 2011-2016 - pancake */
|
2011-09-09 09:49:55 +02:00
|
|
|
|
|
|
|
#include <r_core.h>
|
|
|
|
|
2014-09-03 23:00:38 +02:00
|
|
|
R_API int r_core_patch_line (RCore *core, char *str) {
|
|
|
|
char *p, *q;
|
2016-10-26 23:16:31 +02:00
|
|
|
p = strchr (str + 1, ' ');
|
|
|
|
if (!p) {
|
2014-09-03 23:00:38 +02:00
|
|
|
return 0;
|
2016-10-26 23:16:31 +02:00
|
|
|
}
|
2014-09-03 23:00:38 +02:00
|
|
|
*p = 0;
|
2016-10-26 23:16:31 +02:00
|
|
|
for (++p; *p == ' '; p++); // XXX: skipsspaces here
|
2014-09-03 23:00:38 +02:00
|
|
|
|
|
|
|
switch (*p) {
|
|
|
|
case '"':
|
2016-10-26 23:16:31 +02:00
|
|
|
q = strchr (p + 1,'"');
|
2014-09-03 23:00:38 +02:00
|
|
|
if (q) *q = 0;
|
|
|
|
r_core_cmdf (core, "s %s", str);
|
|
|
|
r_core_cmdf (core, "\"w %s\"", p+1);
|
|
|
|
break;
|
|
|
|
case ':':
|
|
|
|
r_core_cmdf (core, "s %s", str);
|
|
|
|
r_core_cmdf (core, "wa %s", p);
|
|
|
|
break;
|
2017-06-23 03:29:10 +02:00
|
|
|
case 'v':
|
|
|
|
q = strchr (p + 1,' ');
|
|
|
|
if (q) {
|
|
|
|
*q = 0;
|
|
|
|
for (++q; *q == ' '; q++); // XXX: skipsspaces here
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
r_core_cmdf (core, "s %s", str);
|
|
|
|
r_core_cmdf (core, "wv%s %s", p + 1, q);
|
|
|
|
break;
|
2014-09-03 23:00:38 +02:00
|
|
|
default:
|
|
|
|
r_core_cmdf (core, "s %s", str);
|
|
|
|
r_core_cmdf (core, "wx %s", p);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2015-07-08 14:21:43 +02:00
|
|
|
static int __core_patch_bracket(RCore *core, const char *str, ut64 *noff) {
|
|
|
|
char tmp[128];
|
|
|
|
char *s, *p, *q, *off;
|
|
|
|
RBuffer *b = r_buf_new ();
|
2016-10-26 23:16:31 +02:00
|
|
|
if (!b) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
p = off = strdup (str);
|
2015-07-08 14:21:43 +02:00
|
|
|
if (!p) {
|
|
|
|
r_buf_free (b);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
for (;*p;) {
|
|
|
|
if (*p=='\n') {
|
|
|
|
*p++ = 0;
|
|
|
|
} else {
|
|
|
|
p++;
|
|
|
|
continue;
|
|
|
|
}
|
2016-10-26 23:16:31 +02:00
|
|
|
if (*str == '}')
|
2015-07-08 14:21:43 +02:00
|
|
|
break;
|
2016-10-26 23:16:31 +02:00
|
|
|
if ((q = strstr (str, "${"))) {
|
2015-07-08 14:21:43 +02:00
|
|
|
char *end = strchr (q+2,'}');
|
|
|
|
if (end) {
|
|
|
|
*q = *end = 0;
|
|
|
|
*noff = r_num_math (core->num, q+2);
|
|
|
|
r_buf_append_bytes (b, (const ut8*)str, strlen (str));
|
|
|
|
snprintf (tmp, sizeof (tmp), "0x%08"PFMT64x, *noff);
|
|
|
|
r_buf_append_bytes (b, (const ut8*)tmp, strlen (tmp));
|
|
|
|
r_buf_append_bytes (b, (const ut8*)end+1, strlen (end+1));
|
|
|
|
}
|
2016-10-26 23:16:31 +02:00
|
|
|
} else {
|
|
|
|
r_buf_append_bytes (b, (const ut8*)str, strlen (str));
|
|
|
|
}
|
2015-07-08 14:21:43 +02:00
|
|
|
str = p;
|
|
|
|
}
|
|
|
|
|
|
|
|
s = r_buf_to_string (b);
|
|
|
|
r_egg_load (core->egg, s, 0);
|
|
|
|
free (s);
|
|
|
|
|
2017-10-02 21:31:31 +08:00
|
|
|
(void)r_egg_compile (core->egg);
|
2017-08-25 18:12:12 +02:00
|
|
|
(void)r_egg_assemble (core->egg);
|
2015-07-08 14:21:43 +02:00
|
|
|
|
|
|
|
r_buf_free (b);
|
|
|
|
b = r_egg_get_bin (core->egg);
|
|
|
|
|
2016-10-26 23:16:31 +02:00
|
|
|
if (strcmp (off, "+")) {
|
2015-07-08 14:21:43 +02:00
|
|
|
*noff = r_num_math (core->num, off);
|
2016-10-26 23:16:31 +02:00
|
|
|
}
|
2015-07-08 14:21:43 +02:00
|
|
|
r_core_write_at (core, *noff, b->buf, b->length);
|
|
|
|
*noff += b->length;
|
|
|
|
free (off);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2011-09-09 09:49:55 +02:00
|
|
|
R_API int r_core_patch (RCore *core, const char *patch) {
|
2014-09-03 23:00:38 +02:00
|
|
|
char *p, *p0, *str;
|
2011-11-13 05:26:07 +01:00
|
|
|
ut64 noff = 0LL;
|
2011-09-09 09:49:55 +02:00
|
|
|
|
2014-09-03 23:00:38 +02:00
|
|
|
p = p0 = str = strdup (patch);
|
2016-10-26 23:16:31 +02:00
|
|
|
if (!p) {
|
2014-09-03 23:00:38 +02:00
|
|
|
return 0;
|
2016-10-26 23:16:31 +02:00
|
|
|
}
|
2015-07-08 14:21:43 +02:00
|
|
|
for (; *p; p++) {
|
2014-09-03 23:00:38 +02:00
|
|
|
/* read until newline */
|
2016-10-26 23:16:31 +02:00
|
|
|
if (!*p || *p == '\n') {
|
|
|
|
*p++ = 0;
|
|
|
|
} else {
|
|
|
|
continue;
|
|
|
|
}
|
2014-09-03 23:00:38 +02:00
|
|
|
|
|
|
|
switch (*str) {
|
|
|
|
case '#':
|
|
|
|
case '\n':
|
|
|
|
case '\r':
|
|
|
|
case '\0':
|
|
|
|
break;
|
|
|
|
case ':':
|
2013-09-04 02:31:14 +02:00
|
|
|
r_core_cmd0 (core, str+1);
|
2014-09-03 23:00:38 +02:00
|
|
|
break;
|
|
|
|
case '.':
|
|
|
|
case '!':
|
2011-09-09 09:49:55 +02:00
|
|
|
r_core_cmd0 (core, str);
|
2014-09-03 23:00:38 +02:00
|
|
|
break;
|
2015-07-08 14:21:43 +02:00
|
|
|
case '{':
|
|
|
|
(void)__core_patch_bracket (core, str, &noff);
|
|
|
|
break;
|
2014-05-28 14:57:43 +02:00
|
|
|
default:
|
2014-09-03 23:00:38 +02:00
|
|
|
r_core_patch_line (core, str);
|
2014-05-28 14:57:43 +02:00
|
|
|
break;
|
2011-09-09 09:49:55 +02:00
|
|
|
}
|
2016-10-26 23:16:31 +02:00
|
|
|
if (!*p) {
|
|
|
|
break;
|
|
|
|
}
|
2014-09-03 23:00:38 +02:00
|
|
|
str = p;
|
2011-09-09 09:49:55 +02:00
|
|
|
}
|
2014-09-03 23:00:38 +02:00
|
|
|
// eprintf ("%d\n", *p);
|
|
|
|
// eprintf ("Missing newline\n");
|
|
|
|
free (p0);
|
2011-09-09 09:49:55 +02:00
|
|
|
return 0;
|
|
|
|
}
|