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
75
76
77
78
79
80
81
82
83
84
85
86
87
88
|
use yaxpeax_x86::long_mode::{InstDecoder, Operand, RegSpec};
use yaxpeax_x86::MemoryAccessSize;
#[test]
fn register_widths() {
assert_eq!(Operand::Register { reg: RegSpec::rsp() }.width(), Some(8));
assert_eq!(Operand::Register { reg: RegSpec::esp() }.width(), Some(4));
assert_eq!(Operand::Register { reg: RegSpec::sp() }.width(), Some(2));
assert_eq!(Operand::Register { reg: RegSpec::cl() }.width(), Some(1));
assert_eq!(Operand::Register { reg: RegSpec::ch() }.width(), Some(1));
assert_eq!(Operand::Register { reg: RegSpec::gs() }.width(), Some(2));
}
#[test]
fn memory_widths() {
// the register operand directly doesn't report a size - it comes from the `Instruction` for
// which this is an operand .
assert_eq!(Operand::MemDeref { base: RegSpec::rsp() }.width(), None);
fn mem_size_of(data: &[u8]) -> MemoryAccessSize {
let decoder = InstDecoder::default();
decoder.decode_slice(data).unwrap().mem_size().unwrap()
}
// and checking the memory size direcly reports correct names
assert_eq!(mem_size_of(&[0x32, 0x00]).size_name(), "byte");
assert_eq!(mem_size_of(&[0x66, 0x33, 0x00]).size_name(), "word");
assert_eq!(mem_size_of(&[0x33, 0x00]).size_name(), "dword");
assert_eq!(mem_size_of(&[0x48, 0x33, 0x00]).size_name(), "qword");
assert_eq!(mem_size_of(&[0x6a, 0x00]).size_name(), "qword");
assert_eq!(mem_size_of(&[0x68, 0x00, 0x00, 0x00, 0x00]).size_name(), "qword");
// "maskmovq mm0, mm1"
assert_eq!(mem_size_of(&[0x4f, 0x0f, 0xf7, 0xc1]).size_name(), "qword");
assert_eq!(mem_size_of(&[0x67, 0x0f, 0xf7, 0xc1]).size_name(), "dword");
// "maskmovdqu xmm0, xmm1"
assert_eq!(mem_size_of(&[0x66, 0x4f, 0x0f, 0xf7, 0xc1]).size_name(), "qword");
assert_eq!(mem_size_of(&[0x67, 0x66, 0x0f, 0xf7, 0xc1]).size_name(), "dword");
}
#[test]
fn test_implied_memory_width() {
fn mem_size_of(data: &[u8]) -> Option<u8> {
let decoder = InstDecoder::default();
decoder.decode_slice(data).unwrap().mem_size().unwrap().bytes_size()
}
// test push, pop, call, and ret
assert_eq!(mem_size_of(&[0xc3]), Some(8));
assert_eq!(mem_size_of(&[0xe8, 0x11, 0x22, 0x33, 0x44]), Some(8));
assert_eq!(mem_size_of(&[0x50]), Some(8));
assert_eq!(mem_size_of(&[0x58]), Some(8));
assert_eq!(mem_size_of(&[0x66, 0x50]), Some(8));
assert_eq!(mem_size_of(&[0x66, 0x58]), Some(8));
assert_eq!(mem_size_of(&[0xff, 0xf0]), Some(8));
assert_eq!(mem_size_of(&[0x66, 0xff, 0xf0]), Some(2));
// push imm
assert_eq!(mem_size_of(&[0x6a, 0xaa]), Some(8));
assert_eq!(mem_size_of(&[0x68, 0xaa, 0xbb, 0xcc, 0xdd]), Some(8));
// operand-size prefixed call and jump still reads 8 bytes (prefix ignored)
assert_eq!(mem_size_of(&[0x66, 0xff, 0x10]), Some(8));
assert_eq!(mem_size_of(&[0x66, 0xff, 0x20]), Some(8));
// pushf
assert_eq!(mem_size_of(&[0x9c]), Some(8));
// popf
assert_eq!(mem_size_of(&[0x9d]), Some(8));
// leave
assert_eq!(mem_size_of(&[0xc9]), Some(8));
// xlat
assert_eq!(mem_size_of(&[0xd7]), Some(1));
// push fs
assert_eq!(mem_size_of(&[0x0f, 0xa0]), Some(8));
// pop fs
assert_eq!(mem_size_of(&[0x0f, 0xa1]), Some(8));
// push gs
assert_eq!(mem_size_of(&[0x0f, 0xa8]), Some(8));
// pop gs
assert_eq!(mem_size_of(&[0x0f, 0xa9]), Some(8));
// lidt
assert_eq!(mem_size_of(&[0x0f, 0x01, 0x18]), Some(10));
// sidt
assert_eq!(mem_size_of(&[0x0f, 0x01, 0x08]), Some(10));
// lgdt
assert_eq!(mem_size_of(&[0x0f, 0x01, 0x10]), Some(10));
// sgdt
assert_eq!(mem_size_of(&[0x0f, 0x01, 0x00]), Some(10));
}
|