2016-04-03 22:17:57 +00:00
|
|
|
/* radare - LGPL - Copyright 2009-2016 - pancake, nibble */
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <r_types.h>
|
|
|
|
#include <r_util.h>
|
|
|
|
#include <r_asm.h>
|
2009-03-08 23:49:15 +00:00
|
|
|
#include "../config.h"
|
|
|
|
|
2013-06-14 00:51:33 +00:00
|
|
|
R_LIB_VERSION (r_asm);
|
|
|
|
|
2012-06-06 00:17:02 +00:00
|
|
|
static RAsmPlugin *asm_static_plugins[] = { R_ASM_STATIC_PLUGINS };
|
2009-02-18 00:49:26 +00:00
|
|
|
|
2016-11-03 09:39:20 +00:00
|
|
|
static int code_align = 0;
|
|
|
|
|
2012-08-22 16:02:23 +00:00
|
|
|
static int r_asm_pseudo_align(RAsmOp *op, char *input) {
|
2016-11-03 09:39:20 +00:00
|
|
|
code_align = r_num_math (NULL, input);
|
2010-09-13 23:29:09 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-08-22 16:02:23 +00:00
|
|
|
static int r_asm_pseudo_string(RAsmOp *op, char *input, int zero) {
|
2016-08-10 16:49:44 +00:00
|
|
|
int len = strlen (input) - 1;
|
|
|
|
if (len < 1) {
|
|
|
|
return 0;
|
|
|
|
}
|
2010-09-15 11:10:10 +00:00
|
|
|
// TODO: if not starting with '"'.. give up
|
2016-08-10 16:49:44 +00:00
|
|
|
if (input[len] == '"') {
|
2010-09-15 11:10:10 +00:00
|
|
|
input[len] = 0;
|
2016-08-10 16:49:44 +00:00
|
|
|
}
|
|
|
|
if (*input == '"') {
|
2010-09-15 11:10:10 +00:00
|
|
|
input++;
|
2016-08-10 16:49:44 +00:00
|
|
|
}
|
2014-02-23 00:56:26 +00:00
|
|
|
len = r_str_unescape (input)+zero;
|
2011-02-24 15:50:29 +00:00
|
|
|
r_hex_bin2str ((ut8*)input, len, op->buf_hex);
|
2014-04-25 21:23:44 +00:00
|
|
|
strncpy ((char*)op->buf, input, R_ASM_BUFSIZE-1);
|
2009-04-14 23:19:42 +00:00
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2010-05-25 22:59:10 +00:00
|
|
|
static inline int r_asm_pseudo_arch(RAsm *a, char *input) {
|
2010-02-03 13:34:00 +00:00
|
|
|
if (!r_asm_use (a, input)) {
|
|
|
|
eprintf ("Error: Unknown plugin\n");
|
2009-09-24 10:29:05 +00:00
|
|
|
return -1;
|
2009-04-19 18:09:07 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-05-25 22:59:10 +00:00
|
|
|
static inline int r_asm_pseudo_bits(RAsm *a, char *input) {
|
2012-08-22 16:02:23 +00:00
|
|
|
if (!(r_asm_set_bits (a, r_num_math (NULL, input)))) {
|
2010-02-03 13:34:00 +00:00
|
|
|
eprintf ("Error: Unsupported bits value\n");
|
2012-08-22 16:02:23 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
2009-04-22 14:29:30 +00:00
|
|
|
}
|
|
|
|
|
2010-05-25 22:59:10 +00:00
|
|
|
static inline int r_asm_pseudo_org(RAsm *a, char *input) {
|
2010-03-14 04:16:35 +00:00
|
|
|
r_asm_set_pc (a, r_num_math (NULL, input));
|
|
|
|
return 0;
|
2009-04-21 22:42:18 +00:00
|
|
|
}
|
|
|
|
|
2012-08-22 16:02:23 +00:00
|
|
|
static inline int r_asm_pseudo_hex(RAsmOp *op, char *input) {
|
2011-02-24 15:50:29 +00:00
|
|
|
int len = r_hex_str2bin (input, op->buf);
|
2016-06-10 01:12:07 +00:00
|
|
|
strncpy (op->buf_hex, r_str_trim_head_tail (input), R_ASM_BUFSIZE-1);
|
2010-09-13 23:29:09 +00:00
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2012-08-22 16:02:23 +00:00
|
|
|
static inline int r_asm_pseudo_intN(RAsm *a, RAsmOp *op, char *input, int n) {
|
2011-11-12 00:52:31 +00:00
|
|
|
const ut8 *p;
|
|
|
|
short s;
|
|
|
|
int i;
|
|
|
|
long int l;
|
|
|
|
ut64 s64 = r_num_math (NULL, input);
|
2016-11-03 11:01:54 +00:00
|
|
|
if (n != 8 && s64 >> (n * 8)) {
|
2011-11-12 00:52:31 +00:00
|
|
|
eprintf ("int16 Out is out of range\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (n == 2) {
|
|
|
|
s = (short)s64;
|
|
|
|
p = (const ut8*)&s;
|
|
|
|
} else if (n == 4) {
|
|
|
|
i = (int)s64;
|
|
|
|
p = (const ut8*)&i;
|
|
|
|
} else if (n == 8) {
|
|
|
|
l = (long int)s64;
|
|
|
|
p = (const ut8*)&l;
|
2016-11-03 09:39:20 +00:00
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
2016-04-26 09:09:15 +00:00
|
|
|
memcpy (op->buf, p, n);
|
2011-11-12 00:52:31 +00:00
|
|
|
r_hex_bin2str (op->buf, n, op->buf_hex);
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
2012-08-22 16:02:23 +00:00
|
|
|
static inline int r_asm_pseudo_int16(RAsm *a, RAsmOp *op, char *input) {
|
2011-11-12 00:52:31 +00:00
|
|
|
return r_asm_pseudo_intN (a, op, input, 2);
|
|
|
|
}
|
|
|
|
|
2012-08-22 16:02:23 +00:00
|
|
|
static inline int r_asm_pseudo_int32(RAsm *a, RAsmOp *op, char *input) {
|
2011-11-12 00:52:31 +00:00
|
|
|
return r_asm_pseudo_intN (a, op, input, 4);
|
|
|
|
}
|
|
|
|
|
2012-08-22 16:02:23 +00:00
|
|
|
static inline int r_asm_pseudo_int64(RAsm *a, RAsmOp *op, char *input) {
|
2011-11-12 00:52:31 +00:00
|
|
|
return r_asm_pseudo_intN (a, op, input, 8);
|
|
|
|
}
|
|
|
|
|
2012-08-22 16:02:23 +00:00
|
|
|
static inline int r_asm_pseudo_byte(RAsmOp *op, char *input) {
|
2010-09-13 23:29:09 +00:00
|
|
|
int i, len = 0;
|
2012-08-31 09:45:06 +00:00
|
|
|
r_str_replace_char (input, ',', ' ');
|
2010-09-13 23:29:09 +00:00
|
|
|
len = r_str_word_count (input);
|
|
|
|
r_str_word_set0 (input);
|
2016-11-03 09:39:20 +00:00
|
|
|
for (i = 0; i < len; i++) {
|
2012-06-14 15:41:07 +00:00
|
|
|
const char *word = r_str_word_get0 (input, i);
|
2010-09-13 23:29:09 +00:00
|
|
|
int num = (int)r_num_math (NULL, word);
|
2011-02-24 15:50:29 +00:00
|
|
|
op->buf[i] = num;
|
2010-09-13 23:29:09 +00:00
|
|
|
}
|
2011-02-24 15:50:29 +00:00
|
|
|
r_hex_bin2str (op->buf, len, op->buf_hex);
|
2009-04-14 23:19:42 +00:00
|
|
|
return len;
|
2009-04-11 16:49:09 +00:00
|
|
|
}
|
|
|
|
|
2012-08-22 16:02:23 +00:00
|
|
|
static inline int r_asm_pseudo_fill(RAsmOp *op, char *input) {
|
|
|
|
int i, repeat=0, size=0, value=0;
|
2010-09-14 09:22:31 +00:00
|
|
|
sscanf (input, "%d,%d,%d", &repeat, &size, &value); // use r_num?
|
|
|
|
size *= repeat;
|
|
|
|
if (size>0) {
|
|
|
|
for (i=0; i<size; i++)
|
2011-02-24 15:50:29 +00:00
|
|
|
op->buf[i] = value;
|
|
|
|
r_hex_bin2str (op->buf, size, op->buf_hex);
|
2010-09-14 09:22:31 +00:00
|
|
|
} else size = 0;
|
|
|
|
return size;
|
2010-09-13 23:29:09 +00:00
|
|
|
}
|
|
|
|
|
2016-04-03 22:17:57 +00:00
|
|
|
static void plugin_free(RAsmPlugin *p) {
|
|
|
|
if (p && p->fini) {
|
|
|
|
p->fini (NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-02-13 02:00:27 +00:00
|
|
|
R_API RAsm *r_asm_new() {
|
2010-05-20 15:40:58 +00:00
|
|
|
int i;
|
2015-10-04 15:57:32 +00:00
|
|
|
RAsm *a = R_NEW0 (RAsm);
|
2016-09-15 14:21:44 +00:00
|
|
|
if (!a) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2016-09-25 22:45:28 +00:00
|
|
|
a->bits = R_SYS_BITS;
|
2012-06-14 15:41:07 +00:00
|
|
|
a->syntax = R_ASM_SYNTAX_INTEL;
|
2016-04-03 22:17:57 +00:00
|
|
|
a->plugins = r_list_newf ((RListFree)plugin_free);
|
2016-09-15 14:21:44 +00:00
|
|
|
if (!a->plugins) {
|
2015-06-20 11:38:11 +00:00
|
|
|
free (a);
|
|
|
|
return NULL;
|
|
|
|
}
|
2016-09-25 22:45:28 +00:00
|
|
|
for (i = 0; asm_static_plugins[i]; i++) {
|
2016-04-13 11:54:23 +00:00
|
|
|
r_asm_add (a, asm_static_plugins[i]);
|
2010-05-20 15:40:58 +00:00
|
|
|
}
|
|
|
|
return a;
|
2010-01-08 17:25:25 +00:00
|
|
|
}
|
|
|
|
|
2011-11-24 05:06:26 +00:00
|
|
|
R_API int r_asm_setup(RAsm *a, const char *arch, int bits, int big_endian) {
|
|
|
|
int ret = 0;
|
|
|
|
ret |= !r_asm_use (a, arch);
|
|
|
|
ret |= !r_asm_set_bits (a, bits);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-10-05 00:38:37 +00:00
|
|
|
// TODO: spagueti
|
|
|
|
R_API int r_asm_filter_input(RAsm *a, const char *f) {
|
2016-09-15 14:21:44 +00:00
|
|
|
if (!a->ifilter) {
|
2011-10-05 00:38:37 +00:00
|
|
|
a->ifilter = r_parse_new ();
|
2016-09-15 14:21:44 +00:00
|
|
|
}
|
2011-10-05 00:38:37 +00:00
|
|
|
if (!r_parse_use (a->ifilter, f)) {
|
|
|
|
r_parse_free (a->ifilter);
|
|
|
|
a->ifilter = NULL;
|
2015-09-14 00:08:31 +00:00
|
|
|
return false;
|
2011-10-05 00:38:37 +00:00
|
|
|
}
|
2015-09-14 00:08:31 +00:00
|
|
|
return true;
|
2011-10-05 00:38:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_asm_filter_output(RAsm *a, const char *f) {
|
|
|
|
if (!a->ofilter)
|
|
|
|
a->ofilter = r_parse_new ();
|
|
|
|
if (!r_parse_use (a->ofilter, f)) {
|
|
|
|
r_parse_free (a->ofilter);
|
|
|
|
a->ofilter = NULL;
|
2015-09-14 00:08:31 +00:00
|
|
|
return false;
|
2011-10-05 00:38:37 +00:00
|
|
|
}
|
2015-09-14 00:08:31 +00:00
|
|
|
return true;
|
2011-10-05 00:38:37 +00:00
|
|
|
}
|
|
|
|
|
2014-08-29 14:26:43 +00:00
|
|
|
R_API RAsm *r_asm_free(RAsm *a) {
|
|
|
|
if (a) {
|
2014-11-13 23:40:24 +00:00
|
|
|
if (a->cur && a->cur->fini) {
|
2014-11-13 16:21:06 +00:00
|
|
|
a->cur->fini (a->cur->user);
|
|
|
|
}
|
2014-08-29 14:26:43 +00:00
|
|
|
if (a->plugins) {
|
|
|
|
r_list_free (a->plugins);
|
|
|
|
a->plugins = NULL;
|
|
|
|
}
|
2014-11-13 16:21:06 +00:00
|
|
|
free (a->cpu);
|
|
|
|
sdb_free (a->pair);
|
|
|
|
a->pair = NULL;
|
2014-08-29 14:26:43 +00:00
|
|
|
free (a);
|
2012-07-20 15:14:28 +00:00
|
|
|
}
|
2014-08-29 14:26:43 +00:00
|
|
|
return NULL;
|
2009-02-18 00:49:26 +00:00
|
|
|
}
|
2009-02-05 21:08:46 +00:00
|
|
|
|
2010-05-25 22:59:10 +00:00
|
|
|
R_API void r_asm_set_user_ptr(RAsm *a, void *user) {
|
2009-02-18 00:49:26 +00:00
|
|
|
a->user = user;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2016-04-03 22:17:57 +00:00
|
|
|
R_API bool r_asm_add(RAsm *a, RAsmPlugin *foo) {
|
2010-05-25 22:59:10 +00:00
|
|
|
RListIter *iter;
|
2010-05-25 23:42:22 +00:00
|
|
|
RAsmPlugin *h;
|
2009-09-24 10:29:05 +00:00
|
|
|
// TODO: cache foo->name length and use memcmp instead of strcmp
|
2016-09-15 14:21:44 +00:00
|
|
|
if (!foo->name) {
|
2015-09-14 00:08:31 +00:00
|
|
|
return false;
|
2016-09-15 14:21:44 +00:00
|
|
|
}
|
|
|
|
if (foo->init) {
|
2010-02-13 02:00:27 +00:00
|
|
|
foo->init (a->user);
|
2016-09-15 14:21:44 +00:00
|
|
|
}
|
|
|
|
r_list_foreach (a->plugins, iter, h) {
|
|
|
|
if (!strcmp (h->name, foo->name)) {
|
2015-09-14 00:08:31 +00:00
|
|
|
return false;
|
2016-09-15 14:21:44 +00:00
|
|
|
}
|
|
|
|
}
|
2010-05-26 00:55:50 +00:00
|
|
|
r_list_append (a->plugins, foo);
|
2015-09-14 00:08:31 +00:00
|
|
|
return true;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-05-25 22:59:10 +00:00
|
|
|
R_API int r_asm_del(RAsm *a, const char *name) {
|
2009-04-14 23:19:42 +00:00
|
|
|
/* TODO: Implement r_asm_del */
|
2015-09-14 00:08:31 +00:00
|
|
|
return false;
|
2009-04-12 22:46:44 +00:00
|
|
|
}
|
|
|
|
|
2013-11-14 12:52:03 +00:00
|
|
|
|
|
|
|
R_API int r_asm_is_valid(RAsm *a, const char *name) {
|
|
|
|
RAsmPlugin *h;
|
|
|
|
RListIter *iter;
|
2016-09-15 14:21:44 +00:00
|
|
|
if (!name || !*name) {
|
2015-09-14 00:08:31 +00:00
|
|
|
return false;
|
2016-09-15 14:21:44 +00:00
|
|
|
}
|
2013-11-14 12:52:03 +00:00
|
|
|
r_list_foreach (a->plugins, iter, h) {
|
2016-09-15 14:21:44 +00:00
|
|
|
if (!strcmp (h->name, name)) {
|
2015-09-14 00:08:31 +00:00
|
|
|
return true;
|
2016-09-15 14:21:44 +00:00
|
|
|
}
|
2013-11-14 12:52:03 +00:00
|
|
|
}
|
2015-09-14 00:08:31 +00:00
|
|
|
return false;
|
2013-11-14 12:52:03 +00:00
|
|
|
}
|
|
|
|
|
2016-05-30 16:53:32 +00:00
|
|
|
R_API bool r_asm_use_assembler(RAsm *a, const char *name) {
|
|
|
|
RAsmPlugin *h;
|
|
|
|
RListIter *iter;
|
2016-06-02 01:19:31 +00:00
|
|
|
if (a) {
|
|
|
|
if (name && *name) {
|
|
|
|
r_list_foreach (a->plugins, iter, h) {
|
|
|
|
if (h->assemble && !strcmp (h->name, name)) {
|
|
|
|
a->acur = h;
|
|
|
|
return true;
|
|
|
|
}
|
2016-05-30 16:53:32 +00:00
|
|
|
}
|
|
|
|
}
|
2016-06-02 01:19:31 +00:00
|
|
|
a->acur = NULL;
|
2016-05-30 16:53:32 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-09-24 10:29:05 +00:00
|
|
|
// TODO: this can be optimized using r_str_hash()
|
2010-05-25 22:59:10 +00:00
|
|
|
R_API int r_asm_use(RAsm *a, const char *name) {
|
2011-11-12 00:52:31 +00:00
|
|
|
char file[1024];
|
2010-05-25 23:42:22 +00:00
|
|
|
RAsmPlugin *h;
|
2010-05-25 22:59:10 +00:00
|
|
|
RListIter *iter;
|
2016-05-30 16:53:32 +00:00
|
|
|
if (!a || !name) {
|
2015-09-14 00:08:31 +00:00
|
|
|
return false;
|
2016-05-30 16:53:32 +00:00
|
|
|
}
|
|
|
|
r_list_foreach (a->plugins, iter, h) {
|
2016-06-02 00:29:44 +00:00
|
|
|
if (!strcmp (h->name, name) && h->arch) {
|
2011-11-12 00:52:31 +00:00
|
|
|
if (!a->cur || (a->cur && strcmp (a->cur->arch, h->arch))) {
|
|
|
|
//const char *dop = r_config_get (core->config, "dir.opcodes");
|
|
|
|
// TODO: allow configurable path for sdb files
|
|
|
|
snprintf (file, sizeof (file), R_ASM_OPCODES_PATH"/%s.sdb", h->arch);
|
2014-03-07 00:26:11 +00:00
|
|
|
sdb_free (a->pair);
|
|
|
|
a->pair = sdb_new (NULL, file, 0);
|
2011-11-12 00:52:31 +00:00
|
|
|
}
|
2009-02-18 00:49:26 +00:00
|
|
|
a->cur = h;
|
2015-09-14 00:08:31 +00:00
|
|
|
return true;
|
2009-02-18 00:49:26 +00:00
|
|
|
}
|
2016-05-30 16:53:32 +00:00
|
|
|
}
|
2014-03-07 00:26:11 +00:00
|
|
|
sdb_free (a->pair);
|
2011-11-12 00:52:31 +00:00
|
|
|
a->pair = NULL;
|
2015-09-14 00:08:31 +00:00
|
|
|
return false;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-05-25 22:59:10 +00:00
|
|
|
R_API int r_asm_set_subarch(RAsm *a, const char *name) {
|
2015-09-14 00:08:31 +00:00
|
|
|
int ret = false;
|
2009-04-23 10:31:55 +00:00
|
|
|
if (a->cur && a->cur->set_subarch)
|
|
|
|
ret = a->cur->set_subarch(a, name);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-05-25 23:42:22 +00:00
|
|
|
static int has_bits(RAsmPlugin *h, int bits) {
|
2013-12-05 17:41:13 +00:00
|
|
|
if (h && h->bits && (bits & h->bits))
|
2015-09-14 00:08:31 +00:00
|
|
|
return true;
|
|
|
|
return false;
|
2009-04-11 21:22:20 +00:00
|
|
|
}
|
|
|
|
|
2013-06-15 00:56:25 +00:00
|
|
|
R_API void r_asm_set_cpu(RAsm *a, const char *cpu) {
|
|
|
|
free (a->cpu);
|
2013-06-17 01:26:48 +00:00
|
|
|
a->cpu = cpu? strdup (cpu): NULL;
|
2013-06-15 00:56:25 +00:00
|
|
|
}
|
|
|
|
|
2010-05-25 22:59:10 +00:00
|
|
|
R_API int r_asm_set_bits(RAsm *a, int bits) {
|
2010-02-13 02:00:27 +00:00
|
|
|
if (has_bits (a->cur, bits)) {
|
2013-12-05 17:41:13 +00:00
|
|
|
a->bits = bits; // TODO : use OR? :)
|
2015-09-14 00:08:31 +00:00
|
|
|
return true;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2015-09-14 00:08:31 +00:00
|
|
|
return false;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2016-04-26 09:09:15 +00:00
|
|
|
R_API bool r_asm_set_big_endian(RAsm *a, bool b) {
|
|
|
|
if (!a || !a->cur) return false;
|
|
|
|
switch (a->cur->endian) {
|
|
|
|
case R_SYS_ENDIAN_NONE:
|
|
|
|
case R_SYS_ENDIAN_BI:
|
|
|
|
// let user select
|
|
|
|
a->big_endian = b;
|
|
|
|
return b;
|
|
|
|
case R_SYS_ENDIAN_LITTLE:
|
|
|
|
a->big_endian = false;
|
|
|
|
return false;
|
|
|
|
case R_SYS_ENDIAN_BIG:
|
|
|
|
a->big_endian = true;
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
eprintf ("RAsmPlugin doesn't specify endianness\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return false;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-05-25 22:59:10 +00:00
|
|
|
R_API int r_asm_set_syntax(RAsm *a, int syntax) {
|
2009-02-05 21:08:46 +00:00
|
|
|
switch (syntax) {
|
2014-06-25 13:44:03 +00:00
|
|
|
case R_ASM_SYNTAX_REGNUM:
|
2010-01-25 10:54:25 +00:00
|
|
|
case R_ASM_SYNTAX_INTEL:
|
2015-08-21 13:29:37 +00:00
|
|
|
case R_ASM_SYNTAX_MASM:
|
2010-01-25 10:54:25 +00:00
|
|
|
case R_ASM_SYNTAX_ATT:
|
2015-05-09 09:11:46 +00:00
|
|
|
case R_ASM_SYNTAX_JZ:
|
2009-02-05 21:08:46 +00:00
|
|
|
a->syntax = syntax;
|
2015-09-14 00:08:31 +00:00
|
|
|
return true;
|
2009-02-05 21:08:46 +00:00
|
|
|
default:
|
2015-09-14 00:08:31 +00:00
|
|
|
return false;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-05-25 22:59:10 +00:00
|
|
|
R_API int r_asm_set_pc(RAsm *a, ut64 pc) {
|
2009-02-05 21:08:46 +00:00
|
|
|
a->pc = pc;
|
2015-09-14 00:08:31 +00:00
|
|
|
return true;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2013-04-09 20:54:04 +00:00
|
|
|
R_API int r_asm_disassemble(RAsm *a, RAsmOp *op, const ut8 *buf, int len) {
|
2016-10-26 22:54:48 +00:00
|
|
|
int oplen, ret;
|
2016-10-26 23:42:00 +00:00
|
|
|
if (!a || !buf) { // || !op || !buf) {
|
2016-10-26 22:54:48 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
ret = op->payload = 0;
|
2013-12-11 02:06:51 +00:00
|
|
|
op->size = 4;
|
2016-08-07 15:37:33 +00:00
|
|
|
if (len < 1) {
|
2014-11-03 02:27:22 +00:00
|
|
|
return 0;
|
2016-08-07 15:37:33 +00:00
|
|
|
}
|
2015-08-30 22:20:07 +00:00
|
|
|
op->buf_asm[0] = '\0';
|
2015-10-14 00:11:53 +00:00
|
|
|
if (a->pcalign) {
|
2016-09-19 13:47:19 +00:00
|
|
|
const int mod = a->pc % a->pcalign;
|
2016-09-15 14:21:44 +00:00
|
|
|
if (mod) {
|
|
|
|
op->size = a->pcalign - mod;
|
2015-10-14 00:11:53 +00:00
|
|
|
strcpy (op->buf_asm, "unaligned");
|
2015-11-16 21:37:02 +00:00
|
|
|
*op->buf_hex = 0;
|
2016-08-07 15:37:33 +00:00
|
|
|
if ((op->size * 4) >= sizeof (op->buf_hex)) {
|
2016-09-15 14:21:44 +00:00
|
|
|
oplen = (sizeof (op->buf_hex) / 4) - 1;
|
2016-08-07 15:37:33 +00:00
|
|
|
}
|
2015-11-16 21:37:02 +00:00
|
|
|
r_hex_bin2str (buf, op->size, op->buf_hex);
|
2015-10-14 00:11:53 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
2016-05-30 16:53:32 +00:00
|
|
|
if (a->cur && a->cur->disassemble) {
|
2011-10-05 00:38:37 +00:00
|
|
|
ret = a->cur->disassemble (a, op, buf, len);
|
2016-05-30 16:53:32 +00:00
|
|
|
}
|
2016-08-10 16:49:44 +00:00
|
|
|
if (ret < 0) {
|
|
|
|
ret = 0;
|
|
|
|
}
|
2013-12-11 01:18:39 +00:00
|
|
|
oplen = r_asm_op_get_size (op);
|
2013-12-11 02:06:51 +00:00
|
|
|
oplen = op->size;
|
2016-09-15 14:21:44 +00:00
|
|
|
if (oplen > len) {
|
|
|
|
oplen = len;
|
|
|
|
}
|
|
|
|
if (oplen < 1) {
|
|
|
|
oplen = 1;
|
|
|
|
}
|
|
|
|
if (!op->buf_asm[0] || op->size < 1 || !strcmp (op->buf_asm, "invalid")) {
|
2015-08-30 22:20:07 +00:00
|
|
|
if (a->invhex) {
|
2015-09-16 20:20:14 +00:00
|
|
|
if (a->bits == 16) {
|
2016-04-26 09:09:15 +00:00
|
|
|
ut16 b = r_read_le16 (buf);
|
2015-09-16 20:20:14 +00:00
|
|
|
snprintf (op->buf_asm, sizeof (op->buf_asm), ".word 0x%04x", b);
|
2015-09-15 15:02:45 +00:00
|
|
|
} else {
|
2016-04-26 09:09:15 +00:00
|
|
|
ut32 b = r_read_le32 (buf);
|
2015-09-16 20:20:14 +00:00
|
|
|
snprintf (op->buf_asm, sizeof (op->buf_asm), ".dword 0x%08x", b);
|
2015-09-15 15:02:45 +00:00
|
|
|
}
|
2015-09-16 20:20:14 +00:00
|
|
|
// TODO: something for 64bits too?
|
2015-08-30 22:20:07 +00:00
|
|
|
} else {
|
|
|
|
strcpy (op->buf_asm, "invalid");
|
|
|
|
}
|
|
|
|
}
|
2016-09-15 14:21:44 +00:00
|
|
|
if (a->ofilter) {
|
2015-08-30 22:20:07 +00:00
|
|
|
r_parse_parse (a->ofilter, op->buf_asm, op->buf_asm);
|
2016-09-15 14:21:44 +00:00
|
|
|
}
|
2016-04-26 09:09:15 +00:00
|
|
|
memcpy (op->buf, buf, oplen);
|
2014-02-22 00:58:40 +00:00
|
|
|
*op->buf_hex = 0;
|
2016-11-03 11:01:54 +00:00
|
|
|
if ((oplen * 4) >= sizeof (op->buf_hex)) {
|
2016-09-15 14:21:44 +00:00
|
|
|
oplen = (sizeof (op->buf_hex) / 4) - 1;
|
|
|
|
}
|
2014-02-24 09:55:15 +00:00
|
|
|
r_hex_bin2str (buf, oplen, op->buf_hex);
|
2009-04-14 13:21:19 +00:00
|
|
|
return ret;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2016-05-02 00:46:01 +00:00
|
|
|
typedef int (*Ase)(RAsm *a, RAsmOp *op, const char *buf);
|
|
|
|
|
|
|
|
static Ase findAssembler(RAsm *a, const char *kw) {
|
|
|
|
Ase ase = NULL;
|
|
|
|
RAsmPlugin *h;
|
|
|
|
RListIter *iter;
|
2016-05-30 16:53:32 +00:00
|
|
|
if (a->acur && a->acur->assemble) {
|
|
|
|
return a->acur->assemble;
|
|
|
|
}
|
2016-05-02 00:46:01 +00:00
|
|
|
r_list_foreach (a->plugins, iter, h) {
|
|
|
|
if (h->arch && h->assemble
|
|
|
|
&& has_bits (h, a->bits)
|
|
|
|
&& !strncmp (a->cur->arch,
|
|
|
|
h->arch,
|
|
|
|
strlen (a->cur->arch))) {
|
|
|
|
if (kw) {
|
|
|
|
if (strstr (h->name, kw)) {
|
|
|
|
return h->assemble;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ase = h->assemble;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ase;
|
|
|
|
}
|
2016-05-30 16:53:32 +00:00
|
|
|
|
2012-08-22 16:02:23 +00:00
|
|
|
R_API int r_asm_assemble(RAsm *a, RAsmOp *op, const char *buf) {
|
2009-04-14 13:21:19 +00:00
|
|
|
int ret = 0;
|
2011-10-05 00:38:37 +00:00
|
|
|
char *b = strdup (buf);
|
2016-07-26 16:46:48 +00:00
|
|
|
if (!b) {
|
|
|
|
return 0;
|
|
|
|
}
|
2016-05-30 16:53:32 +00:00
|
|
|
if (a->ifilter) {
|
2011-10-05 00:38:37 +00:00
|
|
|
r_parse_parse (a->ifilter, buf, b);
|
2016-05-30 16:53:32 +00:00
|
|
|
}
|
2012-11-08 08:49:27 +00:00
|
|
|
r_str_case (b, 0); // to-lower
|
2012-08-22 16:02:23 +00:00
|
|
|
memset (op, 0, sizeof (RAsmOp));
|
2009-04-11 21:22:20 +00:00
|
|
|
if (a->cur) {
|
2016-05-02 00:46:01 +00:00
|
|
|
Ase ase = NULL;
|
2010-05-25 22:59:10 +00:00
|
|
|
if (!a->cur->assemble) {
|
|
|
|
/* find callback if no assembler support in current plugin */
|
2016-05-02 00:46:01 +00:00
|
|
|
ase = findAssembler (a, ".ks");
|
|
|
|
if (!ase) {
|
|
|
|
ase = findAssembler (a, ".nz");
|
|
|
|
if (!ase) {
|
|
|
|
ase = findAssembler (a, NULL);
|
2010-05-25 22:59:10 +00:00
|
|
|
}
|
2009-04-13 22:47:02 +00:00
|
|
|
}
|
2015-08-24 15:28:13 +00:00
|
|
|
} else {
|
|
|
|
ase = a->cur->assemble;
|
|
|
|
}
|
|
|
|
if (ase) {
|
|
|
|
ret = ase (a, op, b);
|
|
|
|
}
|
2009-04-11 21:22:20 +00:00
|
|
|
}
|
2011-02-24 15:50:29 +00:00
|
|
|
if (op && ret > 0) {
|
|
|
|
r_hex_bin2str (op->buf, ret, op->buf_hex);
|
2013-12-06 04:18:57 +00:00
|
|
|
op->size = ret;
|
2011-10-05 00:38:37 +00:00
|
|
|
op->buf_hex[ret*2] = 0;
|
2014-04-25 21:34:48 +00:00
|
|
|
strncpy (op->buf_asm, b, R_ASM_BUFSIZE-1);
|
2009-04-14 23:19:42 +00:00
|
|
|
}
|
2012-08-08 15:19:48 +00:00
|
|
|
free (b);
|
2009-04-14 13:21:19 +00:00
|
|
|
return ret;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2009-04-08 23:03:49 +00:00
|
|
|
|
2013-08-15 15:23:26 +00:00
|
|
|
R_API RAsmCode* r_asm_mdisassemble(RAsm *a, const ut8 *buf, int len) {
|
2016-02-15 22:51:20 +00:00
|
|
|
RStrBuf *buf_asm;
|
2010-05-25 22:59:10 +00:00
|
|
|
RAsmCode *acode;
|
2010-01-08 17:25:25 +00:00
|
|
|
int ret, slen;
|
2016-09-15 15:30:38 +00:00
|
|
|
ut64 pc = a->pc;
|
2012-08-22 16:02:23 +00:00
|
|
|
RAsmOp op;
|
2009-07-08 11:49:55 +00:00
|
|
|
ut64 idx;
|
2010-01-08 17:25:25 +00:00
|
|
|
|
2016-09-15 14:21:44 +00:00
|
|
|
if (!(acode = r_asm_code_new ())) {
|
2010-01-08 17:25:25 +00:00
|
|
|
return NULL;
|
2016-09-15 14:21:44 +00:00
|
|
|
}
|
|
|
|
if (!(acode->buf = malloc (1 + len))) {
|
2010-06-04 21:47:35 +00:00
|
|
|
return r_asm_code_free (acode);
|
2016-09-15 14:21:44 +00:00
|
|
|
}
|
2010-06-04 21:47:35 +00:00
|
|
|
memcpy (acode->buf, buf, len);
|
2016-09-15 14:21:44 +00:00
|
|
|
if (!(acode->buf_hex = malloc (2 * len+1))) {
|
2013-11-14 12:52:03 +00:00
|
|
|
return r_asm_code_free (acode);
|
2016-09-15 14:21:44 +00:00
|
|
|
}
|
2011-10-19 17:17:57 +00:00
|
|
|
r_hex_bin2str (buf, len, acode->buf_hex);
|
2016-09-15 14:21:44 +00:00
|
|
|
if (!(acode->buf_asm = malloc (4))) {
|
2011-05-11 17:19:53 +00:00
|
|
|
return r_asm_code_free (acode);
|
2016-09-15 14:21:44 +00:00
|
|
|
}
|
2016-02-15 22:51:20 +00:00
|
|
|
buf_asm = r_strbuf_new (NULL);
|
2016-09-15 15:30:38 +00:00
|
|
|
for (idx = ret = slen = 0, acode->buf_asm[0] = '\0'; idx < len; idx += ret) {
|
|
|
|
r_asm_set_pc (a, pc + idx);
|
2016-09-15 14:21:44 +00:00
|
|
|
ret = r_asm_disassemble (a, &op, buf + idx, len - idx);
|
|
|
|
if (ret < 1) {
|
2015-03-05 23:56:10 +00:00
|
|
|
ret = 1;
|
2010-02-13 02:00:27 +00:00
|
|
|
}
|
2016-09-15 14:21:44 +00:00
|
|
|
if (a->ofilter) {
|
2011-10-05 00:38:37 +00:00
|
|
|
r_parse_parse (a->ofilter, op.buf_asm, op.buf_asm);
|
2016-09-15 14:21:44 +00:00
|
|
|
}
|
2016-02-15 22:51:20 +00:00
|
|
|
r_strbuf_append (buf_asm, op.buf_asm);
|
|
|
|
r_strbuf_append (buf_asm, "\n");
|
2009-04-26 22:34:54 +00:00
|
|
|
}
|
2016-02-15 22:51:20 +00:00
|
|
|
acode->buf_asm = r_strbuf_drain (buf_asm);
|
2010-01-08 18:37:42 +00:00
|
|
|
acode->len = idx;
|
2010-01-08 17:25:25 +00:00
|
|
|
return acode;
|
2009-04-26 22:34:54 +00:00
|
|
|
}
|
|
|
|
|
2010-05-30 11:00:21 +00:00
|
|
|
R_API RAsmCode* r_asm_mdisassemble_hexstr(RAsm *a, const char *hexstr) {
|
|
|
|
RAsmCode *ret;
|
|
|
|
ut8 *buf;
|
|
|
|
int len;
|
2016-11-03 11:01:54 +00:00
|
|
|
if (!(buf = malloc (strlen (hexstr) + 1))) {
|
2010-05-30 11:00:21 +00:00
|
|
|
return NULL;
|
2016-11-03 11:01:54 +00:00
|
|
|
}
|
2010-05-30 11:00:21 +00:00
|
|
|
len = r_hex_str2bin (hexstr, buf);
|
2016-01-23 06:27:00 +00:00
|
|
|
if (len < 1) {
|
2011-10-19 17:17:57 +00:00
|
|
|
free (buf);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
ret = r_asm_mdisassemble (a, buf, (ut64)len);
|
2016-11-03 11:01:54 +00:00
|
|
|
if (ret && a->ofilter) {
|
2011-10-05 00:38:37 +00:00
|
|
|
r_parse_parse (a->ofilter, ret->buf_asm, ret->buf_asm);
|
2016-11-03 11:01:54 +00:00
|
|
|
}
|
2010-05-30 11:00:21 +00:00
|
|
|
free (buf);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-07-06 22:53:08 +00:00
|
|
|
R_API RAsmCode* r_asm_assemble_file(RAsm *a, const char *file) {
|
2016-11-03 11:01:54 +00:00
|
|
|
RAsmCode *ac = NULL;
|
2011-07-06 22:53:08 +00:00
|
|
|
char *f = r_file_slurp (file, NULL);
|
2016-11-03 11:01:54 +00:00
|
|
|
if (f) {
|
|
|
|
ac = r_asm_massemble (a, f);
|
|
|
|
free (f);
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
2011-07-06 22:53:08 +00:00
|
|
|
return ac;
|
|
|
|
}
|
|
|
|
|
2010-05-25 22:59:10 +00:00
|
|
|
R_API RAsmCode* r_asm_massemble(RAsm *a, const char *buf) {
|
2014-02-10 02:31:12 +00:00
|
|
|
int labels = 0, num, stage, ret, idx, ctr, i, j, linenum = 0;
|
2016-11-03 11:01:54 +00:00
|
|
|
char *lbuf = NULL, *ptr2, *ptr = NULL, *ptr_start = NULL;
|
|
|
|
char *tokens[R_ASM_BUFSIZE], buf_token[R_ASM_BUFSIZE];
|
2013-11-03 12:47:34 +00:00
|
|
|
RAsmCode *acode = NULL;
|
2016-04-26 09:09:15 +00:00
|
|
|
RAsmOp op = {0};
|
2013-11-10 03:33:12 +00:00
|
|
|
ut64 off, pc;
|
2016-11-03 09:39:20 +00:00
|
|
|
if (!buf) {
|
2010-02-13 02:00:27 +00:00
|
|
|
return NULL;
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
|
|
|
if (!(acode = r_asm_code_new ())) {
|
2010-02-13 02:00:27 +00:00
|
|
|
return NULL;
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
2016-11-03 11:01:54 +00:00
|
|
|
if (!(acode->buf_asm = malloc (strlen (buf) + 16))) {
|
2010-02-13 02:00:27 +00:00
|
|
|
return r_asm_code_free (acode);
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
2016-11-03 11:01:54 +00:00
|
|
|
strncpy (acode->buf_asm, buf, sizeof (acode->buf_asm) - 1);
|
2016-11-03 09:39:20 +00:00
|
|
|
if (!(acode->buf_hex = malloc (64))) { // WTF unefficient
|
2010-02-13 02:00:27 +00:00
|
|
|
return r_asm_code_free (acode);
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
2011-07-05 23:29:18 +00:00
|
|
|
*acode->buf_hex = 0;
|
2016-11-03 09:39:20 +00:00
|
|
|
if (!(acode->buf = malloc (64))) {
|
2010-02-12 12:45:03 +00:00
|
|
|
return r_asm_code_free (acode);
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
2010-02-12 12:45:03 +00:00
|
|
|
lbuf = strdup (buf);
|
2016-11-03 11:01:54 +00:00
|
|
|
code_align = 0;
|
2014-05-15 00:55:20 +00:00
|
|
|
memset (&op, 0, sizeof (op));
|
2009-04-08 23:03:49 +00:00
|
|
|
|
2013-01-15 08:47:29 +00:00
|
|
|
/* accept ';' as comments when input is multiline */
|
2013-02-24 20:12:30 +00:00
|
|
|
{
|
|
|
|
char *nl = strchr (lbuf, '\n');
|
|
|
|
if (nl) {
|
|
|
|
if (strchr (nl+1, '\n'))
|
|
|
|
r_str_replace_char (lbuf, ';', '#');
|
|
|
|
}
|
|
|
|
}
|
2013-04-21 22:01:41 +00:00
|
|
|
// XXX: ops like mov eax, $pc+33 fail coz '+' is nov alid number!!!
|
|
|
|
// XXX: must be handled here to be global.. and not arch-specific
|
|
|
|
{
|
|
|
|
char val[32];
|
|
|
|
snprintf (val, sizeof (val), "0x%"PFMT64x, a->pc);
|
2013-04-24 07:14:08 +00:00
|
|
|
lbuf = r_str_replace (lbuf, "$$", val, 1);
|
2013-04-21 22:01:41 +00:00
|
|
|
}
|
2013-04-21 23:09:27 +00:00
|
|
|
if (a->syscall) {
|
|
|
|
char val[32];
|
2014-02-10 02:31:12 +00:00
|
|
|
char *aa, *p = strstr (lbuf, "$sys.");
|
|
|
|
while (p) {
|
|
|
|
char *sp = (char*)r_str_closer_chr (p, " \n\r#");
|
|
|
|
if (sp) {
|
|
|
|
char osp = *sp;
|
|
|
|
*sp = 0;
|
|
|
|
aa = strdup (p);
|
|
|
|
*sp = osp;
|
|
|
|
num = r_syscall_get_num (a->syscall, aa+5);
|
|
|
|
snprintf (val, sizeof (val), "%d", num);
|
|
|
|
lbuf = r_str_replace (lbuf, aa, val, 1);
|
|
|
|
free (aa);
|
|
|
|
}
|
|
|
|
p = strstr (p+5, "$sys.");
|
2013-04-21 23:09:27 +00:00
|
|
|
}
|
|
|
|
}
|
2016-11-03 09:39:20 +00:00
|
|
|
if (strchr (lbuf, ':')) {
|
2009-04-16 16:03:51 +00:00
|
|
|
labels = 1;
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
2010-02-13 02:00:27 +00:00
|
|
|
/* Tokenize */
|
2009-04-08 23:03:49 +00:00
|
|
|
for (tokens[0] = lbuf, ctr = 0;
|
2014-12-04 23:42:15 +00:00
|
|
|
ctr < R_ASM_BUFSIZE - 1 &&
|
|
|
|
((ptr = strchr (tokens[ctr], ';')) ||
|
2010-02-13 02:00:27 +00:00
|
|
|
(ptr = strchr (tokens[ctr], '\n')) ||
|
2014-12-04 23:42:15 +00:00
|
|
|
(ptr = strchr (tokens[ctr], '\r')));
|
2013-02-24 20:12:30 +00:00
|
|
|
tokens[++ctr] = ptr+1) {
|
2009-04-08 23:03:49 +00:00
|
|
|
*ptr = '\0';
|
2013-02-24 20:12:30 +00:00
|
|
|
}
|
2009-04-08 23:03:49 +00:00
|
|
|
|
2015-09-02 20:33:45 +00:00
|
|
|
#define isavrseparator(x) ((x)==' '||(x)=='\t'||(x)=='\n'||(x)=='\r'||(x)==' '|| \
|
|
|
|
(x)==','||(x)==';'||(x)=='['||(x)==']'|| \
|
|
|
|
(x)=='('||(x)==')'||(x)=='{'||(x)=='}')
|
|
|
|
|
2013-11-10 03:33:12 +00:00
|
|
|
/* Stage 0-2: Parse labels*/
|
|
|
|
/* Stage 3: Assemble */
|
2015-09-18 19:54:25 +00:00
|
|
|
// XXX: stages must be dynamic. until all equs have been resolved
|
2013-11-10 03:33:12 +00:00
|
|
|
#define STAGES 5
|
|
|
|
pc = a->pc;
|
2016-11-03 09:39:20 +00:00
|
|
|
bool inComment = false;
|
2013-11-10 03:33:12 +00:00
|
|
|
for (stage = 0; stage < STAGES; stage++) {
|
2016-11-03 09:39:20 +00:00
|
|
|
if (stage < 2 && !labels) {
|
2009-04-16 16:03:51 +00:00
|
|
|
continue;
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
|
|
|
inComment = false;
|
2013-11-10 03:33:12 +00:00
|
|
|
r_asm_set_pc (a, pc);
|
2010-03-13 13:21:50 +00:00
|
|
|
for (idx = ret = i = j = 0, off = a->pc, acode->buf_hex[0] = '\0';
|
2013-11-10 03:33:12 +00:00
|
|
|
i <= ctr; i++, idx += ret) {
|
2016-11-03 11:29:08 +00:00
|
|
|
memset (buf_token, 0, R_ASM_BUFSIZE);
|
|
|
|
strncpy (buf_token, tokens[i], R_ASM_BUFSIZE - 1);
|
2016-11-03 09:39:20 +00:00
|
|
|
if (inComment) {
|
|
|
|
if (!strncmp (ptr_start, "*/", 2)) {
|
|
|
|
inComment = false;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
2016-11-03 11:29:08 +00:00
|
|
|
// XXX TODO remove arch-specific hacks
|
2015-09-18 19:54:25 +00:00
|
|
|
if (!strncmp (a->cur->arch, "avr", 3)) {
|
2015-09-02 20:33:45 +00:00
|
|
|
for (ptr_start = buf_token; *ptr_start &&
|
|
|
|
isavrseparator (*ptr_start); ptr_start++);
|
|
|
|
} else {
|
|
|
|
for (ptr_start = buf_token; *ptr_start &&
|
|
|
|
isseparator (*ptr_start); ptr_start++);
|
|
|
|
}
|
2016-11-03 09:39:20 +00:00
|
|
|
if (!strncmp (ptr_start, "/*", 2)) {
|
2016-11-03 11:32:50 +00:00
|
|
|
if (!strstr (ptr_start + 2, "*/")) {
|
|
|
|
inComment = true;
|
|
|
|
}
|
2016-11-03 09:39:20 +00:00
|
|
|
continue;
|
|
|
|
}
|
2010-02-13 02:00:27 +00:00
|
|
|
ptr = strchr (ptr_start, '#'); /* Comments */
|
2016-11-03 09:39:20 +00:00
|
|
|
if (ptr && !R_BETWEEN ('0', ptr[1], '9') && ptr[1]!='-') {
|
2010-09-15 08:50:43 +00:00
|
|
|
*ptr = '\0';
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
2013-11-10 03:33:12 +00:00
|
|
|
r_asm_set_pc (a, a->pc + ret);
|
|
|
|
off = a->pc;
|
2010-02-13 02:00:27 +00:00
|
|
|
ret = 0;
|
2016-11-03 11:01:54 +00:00
|
|
|
if (!*ptr_start) {
|
2010-02-12 12:45:03 +00:00
|
|
|
continue;
|
2016-11-03 11:01:54 +00:00
|
|
|
}
|
2013-10-20 21:32:08 +00:00
|
|
|
linenum ++;
|
2015-09-18 19:54:25 +00:00
|
|
|
/* labels */
|
|
|
|
if (labels && (ptr = strchr (ptr_start, ':'))) {
|
|
|
|
bool is_a_label = true;
|
|
|
|
char *q = ptr_start;
|
|
|
|
while (*q) {
|
2016-11-03 11:01:54 +00:00
|
|
|
if (*q == ' ') {
|
2015-09-18 19:54:25 +00:00
|
|
|
is_a_label = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
q++;
|
|
|
|
}
|
|
|
|
if (is_a_label) {
|
|
|
|
//if (stage != 2) {
|
2015-11-19 10:22:50 +00:00
|
|
|
if (ptr_start[1] != 0 && ptr_start[1] != ' ') {
|
2015-09-18 19:54:25 +00:00
|
|
|
char food[64];
|
|
|
|
*ptr = 0;
|
2016-11-03 11:01:54 +00:00
|
|
|
if (code_align) {
|
|
|
|
off += (code_align - (off % code_align));
|
|
|
|
}
|
2015-09-18 19:54:25 +00:00
|
|
|
snprintf (food, sizeof (food), "0x%"PFMT64x"", off);
|
|
|
|
// TODO: warning when redefined
|
|
|
|
r_asm_code_set_equ (acode, ptr_start, food);
|
|
|
|
}
|
|
|
|
//}
|
|
|
|
ptr_start = ptr + 1;
|
|
|
|
}
|
|
|
|
ptr = ptr_start;
|
2010-03-14 04:16:35 +00:00
|
|
|
}
|
|
|
|
if (*ptr_start == '\0') {
|
2010-02-13 02:00:27 +00:00
|
|
|
ret = 0;
|
2014-04-25 21:23:44 +00:00
|
|
|
continue;
|
2015-09-18 19:54:25 +00:00
|
|
|
}
|
|
|
|
if (*ptr_start == '.') { /* pseudo */
|
2016-05-02 00:46:01 +00:00
|
|
|
/* TODO: move into a separate function */
|
2010-02-13 02:00:27 +00:00
|
|
|
ptr = ptr_start;
|
2014-04-25 21:23:44 +00:00
|
|
|
if (!strncmp (ptr, ".intel_syntax", 13))
|
2010-09-10 11:17:24 +00:00
|
|
|
a->syntax = R_ASM_SYNTAX_INTEL;
|
2016-11-03 09:39:20 +00:00
|
|
|
else if (!strncmp (ptr, ".att_syntax", 10)) {
|
2010-09-10 11:17:24 +00:00
|
|
|
a->syntax = R_ASM_SYNTAX_ATT;
|
2016-11-03 09:39:20 +00:00
|
|
|
} else if (!strncmp (ptr, ".asciz", 6)) {
|
|
|
|
r_str_chop (ptr + 8);
|
2016-11-03 11:01:54 +00:00
|
|
|
ret = r_asm_pseudo_string (&op, ptr + 8, 1);
|
2016-11-03 09:39:20 +00:00
|
|
|
} else if (!strncmp (ptr, ".string ", 8)) {
|
2013-04-21 23:18:21 +00:00
|
|
|
r_str_chop (ptr+8);
|
2016-11-03 11:01:54 +00:00
|
|
|
ret = r_asm_pseudo_string (&op, ptr + 8, 1);
|
2016-11-03 09:39:20 +00:00
|
|
|
} else if (!strncmp (ptr, ".ascii ", 6)) {
|
2016-11-03 11:01:54 +00:00
|
|
|
ret = r_asm_pseudo_string (&op, ptr + 7, 0);
|
2016-11-03 09:39:20 +00:00
|
|
|
} else if (!strncmp (ptr, ".align", 6)) {
|
2016-11-03 11:01:54 +00:00
|
|
|
ret = r_asm_pseudo_align (&op, ptr + 7);
|
2013-10-04 11:57:49 +00:00
|
|
|
} else if (!strncmp (ptr, ".arm", 4)) {
|
2013-08-18 17:45:54 +00:00
|
|
|
r_asm_use (a, "arm");
|
|
|
|
r_asm_set_bits (a, 32);
|
|
|
|
ret = 0;
|
2013-10-04 11:57:49 +00:00
|
|
|
} else if (!strncmp (ptr, ".thumb", 6)) {
|
2013-08-18 17:45:54 +00:00
|
|
|
r_asm_use (a, "arm");
|
|
|
|
r_asm_set_bits (a, 16);
|
|
|
|
ret = 0;
|
2013-10-04 11:57:49 +00:00
|
|
|
} else if (!strncmp (ptr, ".arch ", 6))
|
2010-02-12 12:45:03 +00:00
|
|
|
ret = r_asm_pseudo_arch (a, ptr+6);
|
2013-10-04 11:57:49 +00:00
|
|
|
else if (!strncmp (ptr, ".bits ", 6))
|
2010-02-12 12:45:03 +00:00
|
|
|
ret = r_asm_pseudo_bits (a, ptr+6);
|
2013-10-04 11:57:49 +00:00
|
|
|
else if (!strncmp (ptr, ".fill ", 6))
|
2011-02-24 15:50:29 +00:00
|
|
|
ret = r_asm_pseudo_fill (&op, ptr+6);
|
2014-02-10 02:31:12 +00:00
|
|
|
else if (!strncmp (ptr, ".kernel ", 8))
|
|
|
|
r_syscall_setup (a->syscall, a->cur->arch, ptr+8, a->bits);
|
|
|
|
else if (!strncmp (ptr, ".os ", 4))
|
|
|
|
r_syscall_setup (a->syscall, a->cur->arch, ptr+4, a->bits);
|
2013-10-04 11:57:49 +00:00
|
|
|
else if (!strncmp (ptr, ".hex ", 5))
|
2011-02-24 15:50:29 +00:00
|
|
|
ret = r_asm_pseudo_hex (&op, ptr+5);
|
2013-10-04 11:57:49 +00:00
|
|
|
else if ((!strncmp (ptr, ".int16 ", 7)) || !strncmp (ptr, ".short ", 7))
|
2011-11-12 00:52:31 +00:00
|
|
|
ret = r_asm_pseudo_int16 (a, &op, ptr+7);
|
2013-10-04 11:57:49 +00:00
|
|
|
else if (!strncmp (ptr, ".int32 ", 7))
|
2011-11-12 00:52:31 +00:00
|
|
|
ret = r_asm_pseudo_int32 (a, &op, ptr+7);
|
2013-10-04 11:57:49 +00:00
|
|
|
else if (!strncmp (ptr, ".int64 ", 7))
|
2011-11-12 00:52:31 +00:00
|
|
|
ret = r_asm_pseudo_int64 (a, &op, ptr+7);
|
2013-10-04 11:57:49 +00:00
|
|
|
else if (!strncmp (ptr, ".size", 5))
|
2015-09-14 00:08:31 +00:00
|
|
|
ret = true; // do nothing, ignored
|
2013-10-04 11:57:49 +00:00
|
|
|
else if (!strncmp (ptr, ".section", 8))
|
2015-09-14 00:08:31 +00:00
|
|
|
ret = true; // do nothing, ignored
|
2013-10-04 11:57:49 +00:00
|
|
|
else if ((!strncmp (ptr, ".byte ", 6)) || (!strncmp (ptr, ".int8 ", 6)))
|
2011-02-24 15:50:29 +00:00
|
|
|
ret = r_asm_pseudo_byte (&op, ptr+6);
|
2013-10-04 11:57:49 +00:00
|
|
|
else if (!strncmp (ptr, ".glob", 5)) { // .global .globl
|
2010-02-15 21:59:26 +00:00
|
|
|
// eprintf (".global directive not yet implemented\n");
|
|
|
|
ret = 0;
|
|
|
|
continue;
|
2013-10-04 11:57:49 +00:00
|
|
|
} else if (!strncmp (ptr, ".equ ", 5)) {
|
2016-11-03 11:01:54 +00:00
|
|
|
ptr2 = strchr (ptr + 5, ',');
|
2014-04-06 21:20:25 +00:00
|
|
|
if (!ptr2)
|
2016-11-03 11:01:54 +00:00
|
|
|
ptr2 = strchr (ptr + 5, '=');
|
2014-04-06 21:20:25 +00:00
|
|
|
if (!ptr2)
|
2016-11-03 11:01:54 +00:00
|
|
|
ptr2 = strchr (ptr + 5, ' ');
|
2010-02-12 12:45:03 +00:00
|
|
|
if (ptr2) {
|
|
|
|
*ptr2 = '\0';
|
2016-11-03 11:01:54 +00:00
|
|
|
r_asm_code_set_equ (acode, ptr + 5, ptr2 + 1);
|
|
|
|
} else {
|
|
|
|
eprintf ("Invalid syntax for '.equ': Use '.equ <word> <word>'\n");
|
|
|
|
}
|
2013-10-04 11:57:49 +00:00
|
|
|
} else if (!strncmp (ptr, ".org ", 5)) {
|
2010-02-12 12:45:03 +00:00
|
|
|
ret = r_asm_pseudo_org (a, ptr+5);
|
2010-02-13 02:00:27 +00:00
|
|
|
off = a->pc;
|
2013-10-04 11:57:49 +00:00
|
|
|
} else if (!strncmp (ptr, ".text", 5)) {
|
2011-08-07 22:46:04 +00:00
|
|
|
acode->code_offset = a->pc;
|
2013-10-04 11:57:49 +00:00
|
|
|
} else if (!strncmp (ptr, ".data", 5)) {
|
2011-08-07 22:46:04 +00:00
|
|
|
acode->data_offset = a->pc;
|
2010-02-13 02:00:27 +00:00
|
|
|
} else {
|
2011-11-12 00:52:31 +00:00
|
|
|
eprintf ("Unknown directive (%s)\n", ptr);
|
2010-02-13 02:00:27 +00:00
|
|
|
return r_asm_code_free (acode);
|
|
|
|
}
|
2016-11-03 09:39:20 +00:00
|
|
|
if (!ret) {
|
2010-05-25 22:59:10 +00:00
|
|
|
continue;
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
2010-05-25 22:59:10 +00:00
|
|
|
if (ret < 0) {
|
2010-02-13 02:00:27 +00:00
|
|
|
eprintf ("!!! Oops\n");
|
2010-02-12 12:45:03 +00:00
|
|
|
return r_asm_code_free (acode);
|
2010-02-13 02:00:27 +00:00
|
|
|
}
|
2009-04-19 18:09:07 +00:00
|
|
|
} else { /* Instruction */
|
2011-10-05 00:38:37 +00:00
|
|
|
char *str = ptr_start;
|
|
|
|
ptr_start = r_str_chop (str);
|
2016-11-03 09:39:20 +00:00
|
|
|
if (a->ifilter) {
|
2011-10-05 00:38:37 +00:00
|
|
|
r_parse_parse (a->ifilter, ptr_start, ptr_start);
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
2010-02-12 12:45:03 +00:00
|
|
|
if (acode->equs) {
|
2016-11-03 09:39:20 +00:00
|
|
|
if (!*ptr_start) {
|
2011-10-05 00:38:37 +00:00
|
|
|
continue;
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
2011-10-05 00:38:37 +00:00
|
|
|
str = r_asm_code_equ_replace (acode, strdup (ptr_start));
|
2011-02-24 15:50:29 +00:00
|
|
|
ret = r_asm_assemble (a, &op, str);
|
2010-02-12 12:45:03 +00:00
|
|
|
free (str);
|
2011-10-05 00:38:37 +00:00
|
|
|
} else {
|
2016-11-03 09:39:20 +00:00
|
|
|
if (!*ptr_start) {
|
2011-10-05 00:38:37 +00:00
|
|
|
continue;
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
2011-10-05 00:38:37 +00:00
|
|
|
ret = r_asm_assemble (a, &op, ptr_start);
|
|
|
|
}
|
2009-04-19 18:09:07 +00:00
|
|
|
}
|
2016-11-03 09:39:20 +00:00
|
|
|
if (stage == STAGES - 1) {
|
2010-03-13 15:35:18 +00:00
|
|
|
if (ret < 1) {
|
2013-10-20 21:32:08 +00:00
|
|
|
eprintf ("Cannot assemble '%s' at line %d\n", ptr_start, linenum);
|
2010-03-13 15:35:18 +00:00
|
|
|
return r_asm_code_free (acode);
|
2010-02-13 02:00:27 +00:00
|
|
|
}
|
2010-05-25 22:59:10 +00:00
|
|
|
acode->len = idx + ret;
|
2016-11-03 10:11:50 +00:00
|
|
|
char *newbuf = realloc (acode->buf, (idx + ret) * 2);
|
|
|
|
if (!newbuf) {
|
2010-05-25 22:59:10 +00:00
|
|
|
return r_asm_code_free (acode);
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
2016-11-03 10:11:50 +00:00
|
|
|
acode->buf = (ut8*)newbuf;
|
|
|
|
newbuf = realloc (acode->buf_hex, strlen (acode->buf_hex) + strlen (op.buf_hex) + 1);
|
|
|
|
if (!newbuf) {
|
2010-05-25 22:59:10 +00:00
|
|
|
return r_asm_code_free (acode);
|
2016-11-03 09:39:20 +00:00
|
|
|
}
|
2016-11-03 10:11:50 +00:00
|
|
|
acode->buf_hex = newbuf;
|
|
|
|
memcpy (acode->buf + idx, op.buf, ret);
|
2011-02-24 15:50:29 +00:00
|
|
|
strcat (acode->buf_hex, op.buf_hex);
|
2009-04-15 12:37:18 +00:00
|
|
|
}
|
2009-04-08 23:03:49 +00:00
|
|
|
}
|
|
|
|
}
|
2014-05-09 01:06:05 +00:00
|
|
|
free (lbuf);
|
2010-01-08 17:25:25 +00:00
|
|
|
return acode;
|
2009-04-08 23:03:49 +00:00
|
|
|
}
|
2010-04-08 22:52:38 +00:00
|
|
|
|
|
|
|
R_API int r_asm_modify(RAsm *a, ut8 *buf, int field, ut64 val) {
|
2016-11-03 11:01:54 +00:00
|
|
|
if (a->cur && a->cur->modify) {
|
|
|
|
return a->cur->modify (a, buf, field, val);
|
|
|
|
}
|
|
|
|
return false;
|
2010-04-08 22:52:38 +00:00
|
|
|
}
|
2011-02-17 21:03:30 +00:00
|
|
|
|
2011-02-24 15:50:29 +00:00
|
|
|
R_API char *r_asm_op_get_hex(RAsmOp *op) {
|
|
|
|
return strdup (op->buf_hex);
|
2011-02-17 21:03:30 +00:00
|
|
|
}
|
|
|
|
|
2011-02-24 15:50:29 +00:00
|
|
|
R_API char *r_asm_op_get_asm(RAsmOp *op) {
|
|
|
|
return strdup (op->buf_asm);
|
2011-02-17 21:03:30 +00:00
|
|
|
}
|
2011-06-26 18:29:24 +00:00
|
|
|
|
2012-08-22 16:02:23 +00:00
|
|
|
R_API int r_asm_op_get_size(RAsmOp *op) {
|
2013-12-01 23:43:44 +00:00
|
|
|
int len;
|
2016-11-03 11:01:54 +00:00
|
|
|
if (!op) {
|
|
|
|
return 0;
|
|
|
|
}
|
2013-12-06 04:18:57 +00:00
|
|
|
len = op->size - op->payload;
|
2016-11-03 11:01:54 +00:00
|
|
|
if (len < 1) {
|
|
|
|
len = 1;
|
|
|
|
}
|
2013-09-29 23:14:04 +00:00
|
|
|
return len;
|
2012-08-22 16:02:23 +00:00
|
|
|
}
|
|
|
|
|
2011-06-26 18:29:24 +00:00
|
|
|
R_API int r_asm_get_offset(RAsm *a, int type, int idx) { // link to rbin
|
2016-11-03 11:01:54 +00:00
|
|
|
if (a && a->binb.bin && a->binb.get_offset) {
|
2011-06-26 18:29:24 +00:00
|
|
|
return a->binb.get_offset (a->binb.bin, type, idx);
|
2016-11-03 11:01:54 +00:00
|
|
|
}
|
2011-06-26 18:29:24 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2011-11-12 00:52:31 +00:00
|
|
|
|
|
|
|
R_API char *r_asm_describe(RAsm *a, const char* str) {
|
2016-11-03 11:01:54 +00:00
|
|
|
if (!a->pair) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2015-10-31 23:20:58 +00:00
|
|
|
return sdb_get (a->pair, str, 0);
|
2011-11-12 00:52:31 +00:00
|
|
|
}
|
2014-07-02 22:01:46 +00:00
|
|
|
|
|
|
|
R_API RList* r_asm_get_plugins(RAsm *a) {
|
|
|
|
return a->plugins;
|
|
|
|
}
|
2016-04-29 09:30:38 +00:00
|
|
|
|
|
|
|
/* new simplified API */
|
|
|
|
|
|
|
|
R_API bool r_asm_set_arch(RAsm *a, const char *name, int bits) {
|
|
|
|
if (!r_asm_use (a, name)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return r_asm_set_bits (a, bits);
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API char *r_asm_to_string(RAsm *a, ut64 addr, const ut8 *b, int l) {
|
|
|
|
RAsmCode *code;
|
|
|
|
r_asm_set_pc (a, addr);
|
|
|
|
code = r_asm_mdisassemble (a, b, l);
|
|
|
|
if (code) {
|
|
|
|
char *buf_asm = code->buf_asm;
|
|
|
|
code->buf_asm = NULL;
|
|
|
|
r_asm_code_free (code);
|
|
|
|
return buf_asm;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API ut8 *r_asm_from_string(RAsm *a, ut64 addr, const char *b, int *l) {
|
|
|
|
RAsmCode *code;
|
|
|
|
r_asm_set_pc (a, addr);
|
|
|
|
code = r_asm_massemble (a, b);
|
|
|
|
if (code) {
|
|
|
|
ut8 *buf = code->buf;
|
|
|
|
if (l) *l = code->len;
|
|
|
|
r_asm_code_free (code);
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
2016-10-04 13:01:02 +00:00
|
|
|
|
|
|
|
R_API int r_asm_syntax_from_string(const char *name) {
|
|
|
|
if (!strcmp (name, "regnum")) {
|
|
|
|
return R_ASM_SYNTAX_REGNUM;
|
|
|
|
}
|
|
|
|
if (!strcmp (name, "jz")) {
|
|
|
|
return R_ASM_SYNTAX_JZ;
|
|
|
|
}
|
|
|
|
if (!strcmp (name, "intel")) {
|
|
|
|
return R_ASM_SYNTAX_INTEL;
|
|
|
|
}
|
|
|
|
if (!strcmp (name, "masm")) {
|
|
|
|
return R_ASM_SYNTAX_MASM;
|
|
|
|
}
|
|
|
|
if (!strcmp (name, "att")) {
|
|
|
|
return R_ASM_SYNTAX_ATT;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
2016-10-26 22:54:48 +00:00
|
|
|
|
2016-10-30 11:39:49 +00:00
|
|
|
R_API char *r_asm_mnemonics(RAsm *a, int id, bool json) {
|
2016-10-26 22:54:48 +00:00
|
|
|
if (a && a->cur && a->cur->mnemonics) {
|
2016-10-30 11:39:49 +00:00
|
|
|
return a->cur->mnemonics (a, id, json);
|
2016-10-26 22:54:48 +00:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
2016-11-02 02:49:55 +00:00
|
|
|
|
|
|
|
R_API int r_asm_mnemonics_byname(RAsm *a, const char *name) {
|
|
|
|
if (a && a->cur && a->cur->mnemonics) {
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < 1024; i++) {
|
|
|
|
char *n = a->cur->mnemonics (a, i, false);
|
|
|
|
if (n && !strcmp (n, name)) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
free (n);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|