-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathmodrm.h
74 lines (56 loc) · 2.12 KB
/
modrm.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
#ifndef MODRM_H_
#define MODRM_H_
#include <stdint.h>
#include "emulator.h"
typedef struct
{
uint8_t scale; // 2 bits
uint8_t index; // 3 bits
uint8_t base; // 3 bits
} SIB;
typedef struct
{
/* Mod: 2 bits */
uint8_t mod;
/* REG: 3 bits */
union { // union: different data at the same location
uint8_t opcode; // used to extend opcode (e.g. inc: ff + 000 and dec: ff + 001)
uint8_t reg_index; // used to specify register (target or source)
};
uint8_t rm; // 3 bits
SIB sib; // 1 byte
union {
int8_t disp8;
int16_t disp16;
uint32_t disp32;
};
} ModRM;
ModRM create_modrm();
/*
* Parses ModR/M, SIB and Displacement.
* EIP of Emulator needs to be pointing ModR/M Byte.
*/
void parse_modrm(Emulator *emu, ModRM *modrm);
/* Calcurates the memory address. Exposed for LEA instruction. */
uint32_t calc_memory_address(Emulator *emu, ModRM *modrm);
/* Sets 8-bit value to register or memory depending on Mod of ModR/M byte. */
void set_rm8(Emulator *emu, ModRM *modrm, uint8_t value);
/* Gets 8-bit value from register or memory depending on Mod of ModR/M byte. */
uint8_t get_rm8(Emulator *emu, ModRM *modrm);
void set_rm16(Emulator *emu, ModRM *modrm, uint16_t value);
uint16_t get_rm16(Emulator *emu, ModRM *modrm);
void set_rm32(Emulator *emu, ModRM *modrm, uint32_t value);
uint32_t get_rm32(Emulator *emu, ModRM *modrm);
/* Sets 8-bit value on the register specified by REG of ModR/M. */
void set_r8(Emulator *emu, ModRM *modrm, uint8_t value);
/* Get 8-bit value from the register specified by REG of ModR/M. */
uint8_t get_r8(Emulator *emu, ModRM *modrm);
void set_r16(Emulator *emu, ModRM *modrm, uint16_t value);
uint16_t get_r16(Emulator *emu, ModRM *modrm);
void set_r32(Emulator *emu, ModRM *modrm, uint32_t value);
uint32_t get_r32(Emulator *emu, ModRM *modrm);
/* Sets 16-bit value on the segment register specified by REG of ModR/M. */
void set_seg_r(Emulator *emu, ModRM *modrm, uint16_t value);
/* Get 16-bit value from the segment register specified by REG of ModR/M. */
uint16_t get_seg_r(Emulator *emu, ModRM *modrm);
#endif