-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathmod.rs
103 lines (91 loc) · 2.75 KB
/
mod.rs
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
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
// Certain casts are required only on Windows. Inform Clippy to ignore them.
#![allow(clippy::unnecessary_cast)]
use crate::event::ExecModeType;
use libipt_sys::pt_insn;
use std::convert::TryFrom;
mod class;
mod decoder;
pub use class::*;
pub use decoder::*;
/// A single traced instruction.
#[derive(Debug, Clone, Copy)]
pub struct Insn(pub(crate) pt_insn);
impl Insn {
/// The virtual address in its process.
#[must_use]
pub fn ip(self) -> u64 {
self.0.ip
}
/// The image section identifier for the section containing this instruction.
///
/// A value of zero means that the section did not have an identifier.
#[must_use]
pub fn isid(self) -> i32 {
self.0.isid
}
/// The execution mode.
#[must_use]
pub fn mode(self) -> ExecModeType {
ExecModeType::try_from(self.0.mode as u32).expect(concat!(
"unmatched ExecModeType enum value, ",
"this is a bug in either libipt or the bindings"
))
}
/// A coarse classification.
pub fn class(self) -> Class {
Class::try_from(self.0.iclass as u32).expect(concat!(
"unmatched Class enum value, ",
"this is a bug in either libipt or the bindings"
))
}
/// The size in bytes.
#[must_use]
pub fn raw(&self) -> &[u8] {
&self.0.raw[..self.0.size as usize]
}
/// A collection of flags giving additional information:
///
/// - the instruction was executed speculatively.
#[must_use]
pub fn speculative(self) -> bool {
self.0.speculative() > 0
}
/// A collection of flags giving additional information:
///
/// - this instruction is truncated in its image section.
///
/// It starts in the image section identified by \@isid and continues
/// in one or more other sections.
#[must_use]
pub fn truncated(self) -> bool {
self.0.truncated() > 0
}
}
#[cfg(test)]
mod test {
use super::*;
use libipt_sys::pt_exec_mode_ptem_32bit;
use libipt_sys::pt_insn_class_ptic_call;
#[test]
fn test_insn_props() {
let data: [u8; 15] = [17; 15];
let blk = Insn(pt_insn {
ip: 1,
isid: 2,
mode: pt_exec_mode_ptem_32bit,
iclass: pt_insn_class_ptic_call,
raw: data,
size: 8,
_bitfield_align_1: [],
_bitfield_1: pt_insn::new_bitfield_1(0, 1),
__bindgen_padding_0: Default::default(),
});
assert_eq!(blk.ip(), 1);
assert_eq!(blk.isid(), 2);
assert_eq!(blk.mode(), ExecModeType::Bit32);
assert_eq!(blk.class(), Class::Call);
assert_eq!(blk.raw(), &data[..8]);
assert!(blk.truncated());
assert!(!blk.speculative());
}
}