2009-02-05 21:08:46 +00:00
|
|
|
/* radare - LGPL - Copyright 2009 nibble<.ds@gmail.com> */
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#include <r_types.h>
|
|
|
|
#include <r_util.h>
|
|
|
|
#include <r_asm.h>
|
2009-02-18 00:49:26 +00:00
|
|
|
#include <list.h>
|
2009-03-08 23:49:15 +00:00
|
|
|
#include "../config.h"
|
|
|
|
|
|
|
|
static struct r_asm_handle_t *asm_static_plugins[] =
|
2009-03-09 12:03:42 +00:00
|
|
|
{ R_ASM_STATIC_PLUGINS };
|
2009-02-18 00:49:26 +00:00
|
|
|
|
2009-04-22 22:09:55 +00:00
|
|
|
static int r_asm_pseudo_string(struct r_asm_aop_t *aop, char *input)
|
2009-04-14 23:19:42 +00:00
|
|
|
{
|
|
|
|
int len = 0;
|
|
|
|
char *arg = strchr(input, ' ');
|
2009-04-16 16:03:51 +00:00
|
|
|
if (arg && (len = strlen(arg+1))) {
|
|
|
|
arg += 1; len += 1;
|
2009-07-08 11:49:55 +00:00
|
|
|
r_hex_bin2str((ut8*)arg, len, aop->buf_hex);
|
2009-04-19 18:09:07 +00:00
|
|
|
strncpy((char*)aop->buf, arg, R_ASM_BUFSIZE);
|
2009-04-14 23:19:42 +00:00
|
|
|
}
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2009-04-22 22:09:55 +00:00
|
|
|
static int r_asm_pseudo_arch(struct r_asm_t *a, char *input)
|
2009-04-19 18:09:07 +00:00
|
|
|
{
|
|
|
|
char *arg = strchr(input, ' '), str[R_ASM_BUFSIZE];
|
|
|
|
if (arg) {
|
2009-04-22 14:29:30 +00:00
|
|
|
sprintf(str, "asm_%s", arg+1);
|
2009-04-19 18:09:07 +00:00
|
|
|
if (!r_asm_set(a, str)) {
|
|
|
|
fprintf(stderr, "Error: Unknown plugin\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-04-22 22:09:55 +00:00
|
|
|
static int r_asm_pseudo_bits(struct r_asm_t *a, char *input)
|
2009-04-21 22:42:18 +00:00
|
|
|
{
|
|
|
|
char *arg = strchr(input, ' ');
|
2009-04-22 14:29:30 +00:00
|
|
|
if (arg)
|
|
|
|
if (!(r_asm_set_bits(a, r_num_math(NULL, arg+1)))) {
|
|
|
|
fprintf(stderr, "Error: Unsupported bits value\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-04-22 22:09:55 +00:00
|
|
|
static int r_asm_pseudo_org(struct r_asm_t *a, char *input)
|
2009-04-22 14:29:30 +00:00
|
|
|
{
|
|
|
|
char *arg = strchr(input, ' ');
|
|
|
|
if (arg)
|
|
|
|
r_asm_set_pc(a, r_num_math(NULL, arg+1));
|
2009-04-21 22:42:18 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-04-22 22:09:55 +00:00
|
|
|
static int r_asm_pseudo_byte(struct r_asm_aop_t *aop, char *input)
|
2009-04-11 16:49:09 +00:00
|
|
|
{
|
2009-04-14 23:19:42 +00:00
|
|
|
int len = 0;
|
2009-04-11 16:49:09 +00:00
|
|
|
char *arg = strchr(input, ' ');
|
2009-04-14 23:19:42 +00:00
|
|
|
if (arg) {
|
|
|
|
arg += 1;
|
2009-04-19 18:09:07 +00:00
|
|
|
len = r_hex_str2bin(arg, aop->buf);
|
|
|
|
strncpy(aop->buf_hex, r_str_trim(arg), R_ASM_BUFSIZE);
|
2009-04-14 23:19:42 +00:00
|
|
|
}
|
|
|
|
return len;
|
2009-04-11 16:49:09 +00:00
|
|
|
}
|
|
|
|
|
2009-04-07 00:26:41 +00:00
|
|
|
R_API struct r_asm_t *r_asm_new()
|
2009-02-18 00:49:26 +00:00
|
|
|
{
|
|
|
|
struct r_asm_t *a = MALLOC_STRUCT(struct r_asm_t);
|
|
|
|
r_asm_init(a);
|
|
|
|
return a;
|
|
|
|
}
|
|
|
|
|
2009-04-07 00:26:41 +00:00
|
|
|
R_API void r_asm_free(struct r_asm_t *a)
|
2009-02-18 00:49:26 +00:00
|
|
|
{
|
|
|
|
free(a);
|
|
|
|
}
|
2009-02-05 21:08:46 +00:00
|
|
|
|
2009-08-14 00:37:18 +00:00
|
|
|
/* return fastcall register argument 'idx' for a syscall with 'num' args */
|
|
|
|
R_API const char *r_asm_fastcall(struct r_asm_t *a, int idx, int num)
|
|
|
|
{
|
|
|
|
struct r_asm_fastcall_t *fastcall;
|
|
|
|
const char *ret = NULL;
|
|
|
|
int i;
|
|
|
|
if (a && a->cur && a->cur->fastcall)
|
|
|
|
fastcall = *a->cur->fastcall;
|
|
|
|
if (fastcall && idx<=num)
|
|
|
|
for(i=0; 1; i++) {
|
|
|
|
if (i == num) {
|
|
|
|
ret = fastcall[i].arg[idx];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-04-07 00:26:41 +00:00
|
|
|
R_API int r_asm_init(struct r_asm_t *a)
|
2009-02-05 21:08:46 +00:00
|
|
|
{
|
2009-03-08 23:49:15 +00:00
|
|
|
int i;
|
2009-02-18 00:49:26 +00:00
|
|
|
a->user = NULL;
|
2009-03-10 01:49:24 +00:00
|
|
|
a->cur = NULL;
|
2009-04-14 13:21:19 +00:00
|
|
|
a->bits = 32;
|
|
|
|
a->big_endian = 0;
|
|
|
|
a->syntax = R_ASM_SYN_INTEL;
|
|
|
|
a->pc = 0;
|
2009-08-22 01:54:24 +00:00
|
|
|
INIT_LIST_HEAD(&a->asms);
|
2009-03-08 23:49:15 +00:00
|
|
|
for(i=0;asm_static_plugins[i];i++)
|
|
|
|
r_asm_add(a, asm_static_plugins[i]);
|
2009-02-05 21:08:46 +00:00
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2009-04-07 00:26:41 +00:00
|
|
|
R_API void r_asm_set_user_ptr(struct r_asm_t *a, void *user)
|
2009-02-05 21:08:46 +00:00
|
|
|
{
|
2009-02-18 00:49:26 +00:00
|
|
|
a->user = user;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2009-04-07 00:26:41 +00:00
|
|
|
R_API int r_asm_add(struct r_asm_t *a, struct r_asm_handle_t *foo)
|
2009-02-05 21:08:46 +00:00
|
|
|
{
|
2009-03-08 23:49:15 +00:00
|
|
|
struct list_head *pos;
|
2009-02-18 00:49:26 +00:00
|
|
|
if (foo->init)
|
|
|
|
foo->init(a->user);
|
2009-03-08 23:49:15 +00:00
|
|
|
/* avoid dupped plugins */
|
|
|
|
list_for_each_prev(pos, &a->asms) {
|
|
|
|
struct r_asm_handle_t *h = list_entry(pos, struct r_asm_handle_t, list);
|
|
|
|
if (!strcmp(h->name, foo->name))
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
|
2009-02-18 00:49:26 +00:00
|
|
|
list_add_tail(&(foo->list), &(a->asms));
|
|
|
|
return R_TRUE;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2009-04-12 22:46:44 +00:00
|
|
|
R_API int r_asm_del(struct r_asm_t *a, const char *name)
|
|
|
|
{
|
2009-04-14 23:19:42 +00:00
|
|
|
/* TODO: Implement r_asm_del */
|
2009-04-12 22:46:44 +00:00
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
|
2009-04-07 00:26:41 +00:00
|
|
|
R_API int r_asm_list(struct r_asm_t *a)
|
2009-02-05 21:08:46 +00:00
|
|
|
{
|
2009-02-18 00:49:26 +00:00
|
|
|
struct list_head *pos;
|
|
|
|
list_for_each_prev(pos, &a->asms) {
|
|
|
|
struct r_asm_handle_t *h = list_entry(pos, struct r_asm_handle_t, list);
|
|
|
|
printf(" %s: %s\n", h->name, h->desc);
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2009-02-18 00:49:26 +00:00
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
|
2009-04-07 00:26:41 +00:00
|
|
|
R_API int r_asm_set(struct r_asm_t *a, const char *name)
|
2009-02-18 00:49:26 +00:00
|
|
|
{
|
|
|
|
struct list_head *pos;
|
|
|
|
list_for_each_prev(pos, &a->asms) {
|
|
|
|
struct r_asm_handle_t *h = list_entry(pos, struct r_asm_handle_t, list);
|
2009-02-19 15:41:51 +00:00
|
|
|
if (!strcmp(h->name, name)) {
|
2009-02-18 00:49:26 +00:00
|
|
|
a->cur = h;
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return R_FALSE;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2009-04-23 10:31:55 +00:00
|
|
|
R_API int r_asm_set_subarch(struct r_asm_t *a, const char *name)
|
|
|
|
{
|
|
|
|
int ret = R_FALSE;
|
|
|
|
if (a->cur && a->cur->set_subarch)
|
|
|
|
ret = a->cur->set_subarch(a, name);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-04-11 21:22:20 +00:00
|
|
|
static int has_bits(struct r_asm_handle_t *h, int bits)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
if (h && h->bits) {
|
|
|
|
for(i=0; h->bits[i]; i++) {
|
|
|
|
if (bits == h->bits[i])
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-04-07 00:26:41 +00:00
|
|
|
R_API int r_asm_set_bits(struct r_asm_t *a, int bits)
|
2009-02-05 21:08:46 +00:00
|
|
|
{
|
2009-04-14 13:21:19 +00:00
|
|
|
if (has_bits(a->cur, bits)) {
|
2009-02-05 21:08:46 +00:00
|
|
|
a->bits = bits;
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
2009-04-11 21:22:20 +00:00
|
|
|
return R_FALSE;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2009-04-07 00:26:41 +00:00
|
|
|
R_API int r_asm_set_big_endian(struct r_asm_t *a, int boolean)
|
2009-02-05 21:08:46 +00:00
|
|
|
{
|
|
|
|
a->big_endian = boolean;
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2009-04-07 00:26:41 +00:00
|
|
|
R_API int r_asm_set_syntax(struct r_asm_t *a, int syntax)
|
2009-02-05 21:08:46 +00:00
|
|
|
{
|
|
|
|
switch (syntax) {
|
|
|
|
case R_ASM_SYN_INTEL:
|
|
|
|
case R_ASM_SYN_ATT:
|
|
|
|
a->syntax = syntax;
|
|
|
|
return R_TRUE;
|
|
|
|
default:
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-08 11:49:55 +00:00
|
|
|
R_API int r_asm_set_pc(struct r_asm_t *a, ut64 pc)
|
2009-02-05 21:08:46 +00:00
|
|
|
{
|
|
|
|
a->pc = pc;
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2009-07-08 11:49:55 +00:00
|
|
|
R_API int r_asm_disassemble(struct r_asm_t *a, struct r_asm_aop_t *aop, ut8 *buf, ut64 len)
|
2009-02-05 21:08:46 +00:00
|
|
|
{
|
2009-04-14 13:21:19 +00:00
|
|
|
int ret = 0;
|
2009-02-18 00:49:26 +00:00
|
|
|
if (a->cur && a->cur->disassemble)
|
2009-04-14 13:21:19 +00:00
|
|
|
ret = a->cur->disassemble(a, aop, buf, len);
|
|
|
|
if (ret > 0) {
|
|
|
|
memcpy(aop->buf, buf, ret);
|
|
|
|
r_hex_bin2str(buf, ret, aop->buf_hex);
|
|
|
|
}
|
|
|
|
return ret;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2009-04-22 22:09:55 +00:00
|
|
|
R_API int r_asm_assemble(struct r_asm_t *a, struct r_asm_aop_t *aop, char *buf)
|
2009-02-05 21:08:46 +00:00
|
|
|
{
|
2009-04-14 13:21:19 +00:00
|
|
|
int ret = 0;
|
2009-04-11 21:22:20 +00:00
|
|
|
struct list_head *pos;
|
|
|
|
if (a->cur) {
|
|
|
|
if (a->cur->assemble)
|
2009-04-14 13:21:19 +00:00
|
|
|
ret = a->cur->assemble(a, aop, buf);
|
|
|
|
else /* find callback if no assembler support in current plugin */
|
|
|
|
list_for_each_prev(pos, &a->asms) {
|
|
|
|
struct r_asm_handle_t *h = list_entry(pos, struct r_asm_handle_t, list);
|
|
|
|
if (h->arch && h->assemble && has_bits(h, a->bits) && !strcmp(a->cur->arch, h->arch)) {
|
|
|
|
ret = h->assemble(a, aop, buf);
|
|
|
|
break;
|
2009-04-13 22:47:02 +00:00
|
|
|
}
|
|
|
|
}
|
2009-04-11 21:22:20 +00:00
|
|
|
}
|
2009-04-14 23:19:42 +00:00
|
|
|
if (aop && ret > 0) {
|
2009-04-14 13:21:19 +00:00
|
|
|
r_hex_bin2str(aop->buf, ret, aop->buf_hex);
|
2009-04-16 17:20:03 +00:00
|
|
|
strncpy(aop->buf_asm, buf, R_ASM_BUFSIZE);
|
2009-04-14 23:19:42 +00:00
|
|
|
}
|
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
|
|
|
|
2009-07-08 11:49:55 +00:00
|
|
|
R_API ut64 r_asm_mdisassemble(struct r_asm_t *a, struct r_asm_aop_t *aop, ut8 *buf, ut64 len)
|
2009-04-26 22:34:54 +00:00
|
|
|
{
|
|
|
|
int ret;
|
2009-07-08 11:49:55 +00:00
|
|
|
ut64 idx;
|
2009-04-26 22:34:54 +00:00
|
|
|
char buf_asm[R_ASM_BUFSIZE];
|
|
|
|
|
|
|
|
for(idx = ret = 0, buf_asm[0] = '\0'; idx < len; idx+=ret) {
|
|
|
|
r_asm_set_pc(a, a->pc + idx);
|
|
|
|
if (!(ret = r_asm_disassemble(a, aop, buf+idx, len-idx)))
|
|
|
|
return 0;
|
|
|
|
strcat(buf_asm, aop->buf_asm);
|
|
|
|
if (idx + ret < len) strcat(buf_asm, "\n");
|
|
|
|
}
|
|
|
|
strncpy(aop->buf_asm, buf_asm, R_ASM_BUFSIZE);
|
|
|
|
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
2009-04-08 23:03:49 +00:00
|
|
|
R_API int r_asm_massemble(struct r_asm_t *a, struct r_asm_aop_t *aop, char *buf)
|
|
|
|
{
|
2009-04-16 16:03:51 +00:00
|
|
|
struct {
|
2009-05-20 12:41:04 +00:00
|
|
|
char name[32];
|
2009-07-08 11:49:55 +00:00
|
|
|
ut64 offset;
|
2009-05-20 12:41:04 +00:00
|
|
|
} flags[512]; /* XXX: dinamic length */
|
2009-04-16 17:20:03 +00:00
|
|
|
char *lbuf = NULL, *ptr = NULL, *ptr_start = NULL, *label_name = NULL,
|
|
|
|
*tokens[R_ASM_BUFSIZE], buf_hex[R_ASM_BUFSIZE],
|
|
|
|
buf_token[R_ASM_BUFSIZE], buf_token2[R_ASM_BUFSIZE];
|
2009-07-08 11:49:55 +00:00
|
|
|
ut8 buf_bin[R_ASM_BUFSIZE];
|
2009-04-16 16:03:51 +00:00
|
|
|
int labels = 0, stage, ret, idx, ctr, i, j;
|
2009-07-08 11:49:55 +00:00
|
|
|
ut64 label_offset;
|
2009-04-08 23:03:49 +00:00
|
|
|
|
|
|
|
if (buf == NULL)
|
|
|
|
return 0;
|
|
|
|
lbuf = strdup(buf);
|
|
|
|
|
2009-04-16 16:03:51 +00:00
|
|
|
if (strchr(lbuf, '_'))
|
|
|
|
labels = 1;
|
|
|
|
|
2009-04-08 23:03:49 +00:00
|
|
|
for (tokens[0] = lbuf, ctr = 0;
|
2009-04-21 11:39:33 +00:00
|
|
|
(ptr = strchr(tokens[ctr], ';')) ||
|
|
|
|
(ptr = strchr(tokens[ctr], '\n')) || (ptr = strchr(tokens[ctr], '\r'));
|
2009-04-08 23:03:49 +00:00
|
|
|
tokens[++ctr] = ptr+1)
|
|
|
|
*ptr = '\0';
|
|
|
|
|
2009-04-16 16:03:51 +00:00
|
|
|
/* Stage 1: Parse labels*/
|
|
|
|
/* Stage 2: Assemble */
|
|
|
|
for (stage = 0; stage < 2; stage++) {
|
|
|
|
if (stage == 0 && !labels)
|
|
|
|
continue;
|
2009-04-21 10:11:07 +00:00
|
|
|
for (idx = ret = i = j = 0, label_offset = a->pc, buf_hex[0] = '\0';
|
2009-04-19 18:09:07 +00:00
|
|
|
i <= ctr; i++, idx += ret, label_offset += ret) {
|
2009-04-16 17:20:03 +00:00
|
|
|
strncpy(buf_token, tokens[i], R_ASM_BUFSIZE);
|
2009-04-16 16:03:51 +00:00
|
|
|
if (stage == 1)
|
|
|
|
r_asm_set_pc(a, a->pc + ret);
|
2009-04-20 11:26:29 +00:00
|
|
|
for (ptr_start = buf_token;*ptr_start&&isseparator(*ptr_start);ptr_start++);
|
2009-04-16 16:03:51 +00:00
|
|
|
if (labels) { /* Labels */
|
2009-04-19 18:09:07 +00:00
|
|
|
while (ptr_start[0] != '.' && (ptr = strchr(ptr_start, '_'))) {
|
2009-04-16 16:03:51 +00:00
|
|
|
if ((label_name = r_str_word_get_first(ptr))) {
|
|
|
|
if ((ptr == ptr_start)) {
|
2009-04-16 17:20:03 +00:00
|
|
|
if (stage == 0 && j < 1024) {
|
2009-04-16 16:03:51 +00:00
|
|
|
strncpy(flags[j].name, label_name, 256);
|
|
|
|
flags[j].offset = label_offset;
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
ptr_start += strlen(label_name)+1;
|
|
|
|
} else {
|
|
|
|
*ptr = '\0';
|
|
|
|
if (stage == 1) {
|
|
|
|
for (j = 0; j < 1024; j++)
|
|
|
|
if (!strcmp(label_name, flags[j].name)) {
|
|
|
|
label_offset = flags[j].offset;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (j == 1024)
|
|
|
|
return 0;
|
|
|
|
}
|
2009-04-16 17:20:03 +00:00
|
|
|
snprintf(buf_token2, R_ASM_BUFSIZE, "%s0x%llx%s",
|
2009-04-16 16:03:51 +00:00
|
|
|
ptr_start, label_offset, ptr+strlen(label_name));
|
2009-04-16 17:20:03 +00:00
|
|
|
strncpy(buf_token, buf_token2, R_ASM_BUFSIZE);
|
2009-04-16 16:03:51 +00:00
|
|
|
ptr_start = buf_token;
|
|
|
|
}
|
|
|
|
free(label_name);
|
|
|
|
}
|
2009-04-15 12:37:18 +00:00
|
|
|
}
|
2009-04-16 16:03:51 +00:00
|
|
|
}
|
2009-04-19 18:09:07 +00:00
|
|
|
if ((ptr = strchr(ptr_start, '.'))) { /* Pseudo */
|
2009-04-21 22:42:18 +00:00
|
|
|
if (!memcmp(ptr, ".string", 7))
|
2009-04-22 14:29:30 +00:00
|
|
|
ret = r_asm_pseudo_string(aop, ptr);
|
2009-04-21 22:42:18 +00:00
|
|
|
else if (!memcmp(ptr, ".arch", 5))
|
2009-04-22 14:29:30 +00:00
|
|
|
ret = r_asm_pseudo_arch(a, ptr);
|
|
|
|
else if (!memcmp(ptr, ".bits", 5))
|
|
|
|
ret = r_asm_pseudo_bits(a, ptr);
|
2009-04-21 22:42:18 +00:00
|
|
|
else if (!memcmp(ptr, ".byte", 5))
|
2009-04-22 14:29:30 +00:00
|
|
|
ret = r_asm_pseudo_byte(aop, ptr);
|
2009-04-21 22:42:18 +00:00
|
|
|
else if (!memcmp(ptr, ".org", 4))
|
2009-04-22 14:29:30 +00:00
|
|
|
ret = r_asm_pseudo_org(a, ptr);
|
2009-04-19 18:09:07 +00:00
|
|
|
else return 0;
|
|
|
|
if (!ret)
|
|
|
|
continue;
|
|
|
|
else if (ret < 0)
|
|
|
|
return 0;
|
|
|
|
} else { /* Instruction */
|
2009-04-16 16:03:51 +00:00
|
|
|
ret = r_asm_assemble(a, aop, ptr_start);
|
2009-04-19 18:09:07 +00:00
|
|
|
if (!ret)
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (stage == 1) {
|
2009-04-16 17:20:03 +00:00
|
|
|
for (j = 0; j < ret && idx+j < R_ASM_BUFSIZE; j++)
|
2009-04-16 16:03:51 +00:00
|
|
|
buf_bin[idx+j] = aop->buf[j];
|
|
|
|
strcat(buf_hex, aop->buf_hex);
|
2009-04-15 12:37:18 +00:00
|
|
|
}
|
2009-04-08 23:03:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-04-16 17:20:03 +00:00
|
|
|
memcpy(aop->buf, buf_bin, R_ASM_BUFSIZE);
|
|
|
|
memcpy(aop->buf_hex, buf_hex, R_ASM_BUFSIZE);
|
2009-04-08 23:03:49 +00:00
|
|
|
|
|
|
|
return idx;
|
|
|
|
}
|