| #include <stdio.h> |
| #include <stdlib.h> |
| |
| #include "libdis.h" |
| |
| #ifdef _MSC_VER |
| #define snprintf _snprintf |
| #define inline __inline |
| #endif |
| |
| int x86_insn_is_valid( x86_insn_t *insn ) { |
| if ( insn && insn->type != insn_invalid && insn->size > 0 ) { |
| return 1; |
| } |
| |
| return 0; |
| } |
| |
| uint32_t x86_get_address( x86_insn_t *insn ) { |
| x86_oplist_t *op_lst; |
| if (! insn || ! insn->operands ) { |
| return 0; |
| } |
| |
| for (op_lst = insn->operands; op_lst; op_lst = op_lst->next ) { |
| if ( op_lst->op.type == op_offset ) { |
| return op_lst->op.data.offset; |
| } else if ( op_lst->op.type == op_absolute ) { |
| if ( op_lst->op.datatype == op_descr16 ) { |
| return (uint32_t) |
| op_lst->op.data.absolute.offset.off16; |
| } |
| return op_lst->op.data.absolute.offset.off32; |
| } |
| } |
| |
| return 0; |
| } |
| |
| int32_t x86_get_rel_offset( x86_insn_t *insn ) { |
| x86_oplist_t *op_lst; |
| if (! insn || ! insn->operands ) { |
| return 0; |
| } |
| |
| for (op_lst = insn->operands; op_lst; op_lst = op_lst->next ) { |
| if ( op_lst->op.type == op_relative_near ) { |
| return (int32_t) op_lst->op.data.relative_near; |
| } else if ( op_lst->op.type == op_relative_far ) { |
| return op_lst->op.data.relative_far; |
| } |
| } |
| |
| return 0; |
| } |
| |
| x86_op_t * x86_get_branch_target( x86_insn_t *insn ) { |
| x86_oplist_t *op_lst; |
| if (! insn || ! insn->operands ) { |
| return NULL; |
| } |
| |
| for (op_lst = insn->operands; op_lst; op_lst = op_lst->next ) { |
| if ( op_lst->op.access & op_execute ) { |
| return &(op_lst->op); |
| } |
| } |
| |
| return NULL; |
| } |
| x86_op_t * x86_get_imm( x86_insn_t *insn ) { |
| x86_oplist_t *op_lst; |
| if (! insn || ! insn->operands ) { |
| return NULL; |
| } |
| |
| for (op_lst = insn->operands; op_lst; op_lst = op_lst->next ) { |
| if ( op_lst->op.type == op_immediate ) { |
| return &(op_lst->op); |
| } |
| } |
| |
| return NULL; |
| } |
| |
| #define IS_PROPER_IMM( x ) \ |
| x->op.type == op_immediate && ! (x->op.flags & op_hardcode) |
| |
| |
| /* if there is an immediate value in the instruction, return a pointer to |
| * it */ |
| unsigned char * x86_get_raw_imm( x86_insn_t *insn ) { |
| int size, offset; |
| x86_op_t *op = NULL; |
| |
| if (! insn || ! insn->operands ) { |
| return(NULL); |
| } |
| |
| /* a bit inelegant, but oh well... */ |
| if ( IS_PROPER_IMM( insn->operands ) ) { |
| op = &insn->operands->op; |
| } else if ( insn->operands->next ) { |
| if ( IS_PROPER_IMM( insn->operands->next ) ) { |
| op = &insn->operands->next->op; |
| } else if ( insn->operands->next->next && |
| IS_PROPER_IMM( insn->operands->next->next ) ) { |
| op = &insn->operands->next->next->op; |
| } |
| } |
| |
| if (! op ) { |
| return( NULL ); |
| } |
| |
| /* immediate data is at the end of the insn */ |
| size = x86_operand_size( op ); |
| offset = insn->size - size; |
| return( &insn->bytes[offset] ); |
| } |
| |
| |
| unsigned int x86_operand_size( x86_op_t *op ) { |
| switch (op->datatype ) { |
| case op_byte: return 1; |
| case op_word: return 2; |
| case op_dword: return 4; |
| case op_qword: return 8; |
| case op_dqword: return 16; |
| case op_sreal: return 4; |
| case op_dreal: return 8; |
| case op_extreal: return 10; |
| case op_bcd: return 10; |
| case op_ssimd: return 16; |
| case op_dsimd: return 16; |
| case op_sssimd: return 4; |
| case op_sdsimd: return 8; |
| case op_descr32: return 6; |
| case op_descr16: return 4; |
| case op_pdescr32: return 6; |
| case op_pdescr16: return 6; |
| case op_bounds16: return 4; |
| case op_bounds32: return 8; |
| case op_fpuenv16: return 14; |
| case op_fpuenv32: return 28; |
| case op_fpustate16: return 94; |
| case op_fpustate32: return 108; |
| case op_fpregset: return 512; |
| case op_fpreg: return 10; |
| case op_none: return 0; |
| } |
| return(4); /* default size */ |
| } |
| |
| void x86_set_insn_addr( x86_insn_t *insn, uint32_t addr ) { |
| if ( insn ) insn->addr = addr; |
| } |
| |
| void x86_set_insn_offset( x86_insn_t *insn, unsigned int offset ){ |
| if ( insn ) insn->offset = offset; |
| } |
| |
| void x86_set_insn_function( x86_insn_t *insn, void * func ){ |
| if ( insn ) insn->function = func; |
| } |
| |
| void x86_set_insn_block( x86_insn_t *insn, void * block ){ |
| if ( insn ) insn->block = block; |
| } |
| |
| void x86_tag_insn( x86_insn_t *insn ){ |
| if ( insn ) insn->tag = 1; |
| } |
| |
| void x86_untag_insn( x86_insn_t *insn ){ |
| if ( insn ) insn->tag = 0; |
| } |
| |
| int x86_insn_is_tagged( x86_insn_t *insn ){ |
| return insn->tag; |
| } |
| |