forked from hasherezade/tiny_tracer
-
Notifications
You must be signed in to change notification settings - Fork 0
/
FuncWatch.cpp
120 lines (102 loc) · 3.12 KB
/
FuncWatch.cpp
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
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
#include "FuncWatch.h"
#include <vector>
#include <fstream>
#include <sstream>
#include "Util.h"
bool WFuncInfo::load(const std::string &sline, char delimiter)
{
std::vector<std::string> args;
util::splitList(sline, delimiter, args);
if (args.size() < 2) return false;
this->dllName = args[0];
this->funcName = args[1];
this->paramCount = 0;
if (args.size() >= 3) {
this->paramCount = util::loadInt(args[2]);
}
return true;
}
bool WFuncInfo::update(const WFuncInfo &func_info)
{
bool isUpdated = false;
if (this->paramCount < func_info.paramCount) {
this->paramCount = func_info.paramCount;
isUpdated = true;
}
return isUpdated;
}
//---
std::string WSyscallInfo::formatSyscallName(int syscallID)
{
std::stringstream ss;
ss << "SYSCALL:0x" << std::hex << (syscallID);
return ss.str();
}
bool WSyscallInfo::load(const std::string& sline, char delimiter)
{
std::vector<std::string> args;
util::splitList(sline, delimiter, args);
if (args.size() < 3) return false;
// Note: '<' and '>' are used to ensure this cannot overlap with a valid
// file or library name.
if (args[0] != "<SYSCALL>") return false;
// Parse syscall ID as a hexadecimal number
const int syscallId = util::loadInt(args[1], true);
if (syscallId < 0) return false;
this->syscallId = static_cast<uint32_t>(syscallId);
this->paramCount = util::loadInt(args[2]);
return true;
}
bool WSyscallInfo::update(const WSyscallInfo& syscall_info)
{
bool isUpdated = false;
if (this->paramCount < syscall_info.paramCount) {
this->paramCount = syscall_info.paramCount;
isUpdated = true;
}
return isUpdated;
}
//---
void FuncWatchList::appendSyscall(WSyscallInfo& syscall_info)
{
std::map<uint32_t, WSyscallInfo>::iterator it = syscalls.find(syscall_info.syscallId);
if (it == syscalls.end()) {
syscalls[syscall_info.syscallId] = syscall_info;
}
else {
it->second.update(syscall_info);
}
}
size_t FuncWatchList::loadList(const char* filename, FuncList* exclusions)
{
std::ifstream myfile(filename);
if (!myfile.is_open()) {
std::cerr << "Coud not open file: " << filename << std::endl;
return 0;
}
const size_t MAX_LINE = 300;
char line[MAX_LINE] = { 0 };
while (!myfile.eof()) {
myfile.getline(line, MAX_LINE);
const std::string str = line;
if (!str.size() || str[0] == '#') { // skip empty lines and comments
continue;
}
// Try to parse as a syscall
WSyscallInfo syscall_info;
if (syscall_info.load(str, FuncList::DELIM)) {
appendSyscall(syscall_info);
continue;
}
// Try to parse as a function
WFuncInfo func_info;
if (func_info.load(line, FuncList::DELIM)) {
if (exclusions && exclusions->contains(func_info.dllName, func_info.funcName)) {
//std::cout << ">> Skipping: " << func_info.funcName << std::endl;
continue;
}
appendFunc(func_info);
}
}
return funcs.size();
}