3 #include <sys/ptrace.h>
16 #include "Initializer.h"
18 #include "ExitObserver.h"
19 #include "TrapObserver.h"
20 #include "SegfaultObserver.h"
21 #include "MallocObserver.h"
22 #include "FreeObserver.h"
23 #include "ReallocObserver.h"
24 #include "MainObserver.h"
25 #include "BreakpointReference.h"
27 #include "misc/String.h"
35 Portal::Portal(Misc::SmartPointer
<Misc::ArgumentList
> argument_list
) : pid(0), libc_offset(0) {
36 std::cout
<< "Portal::Portal(): called, initialization begun . . . \n";
39 throw PTraceException(Misc::StreamAsString() << "Forking to create child process failed: " << strerror(errno
));
41 ptrace(PTRACE_TRACEME
, 0, NULL
, NULL
);
42 if(execv(argument_list
->get_argument(0).c_str(), argument_list
->get_as_argv()) == -1) {
43 throw PTraceException(Misc::StreamAsString() << "Failed to execute process: " << strerror(errno
));
47 /*ptrace(PTRACE_SETOPTIONS, pid, 0, PTRACE_O_TRACESYSGOOD);*/
49 /* Trap signals are the most common, so add the TrapObserver on first. */
50 add_signal_observer(new TrapObserver());
51 add_signal_observer(new ExitObserver());
52 add_signal_observer(new SegfaultObserver());
54 /* This is a single-shot breakpoint observer, but since there's currently no way to remove observers, stick it on last. */
55 initial_observer
= new MainObserver();
56 malloc_observer
= new MallocObserver();
57 free_observer
= new FreeObserver();
58 realloc_observer
= new ReallocObserver();
60 std::cout
<< "Portal::Portal(): waiting for SIGTRAP call from child . . ." << std::endl
;
61 /* Wait for the SIGTRAP that indicates a exec() call . . . */
63 std::cout
<< "Portal::Portal(): got SIGTRAP call, placing breakpoint on main . . ." << std::endl
;
64 /* place a breakpoint at main, for intialization purposes. */
65 Word main_address
= Initializer::get_instance()->get_program_manager()->get_elf_parser()->get_symbol("main")->get_address();
66 place_breakpoint(main_address
, initial_observer
);
68 std::cout
<< "Portal::Portal(): continuing execution until breakpoint on main . . ." << std::endl
;
69 /* Now continue until main(). */
73 Word
Portal::get_register(ASM::Register which
) const {
74 struct user_regs_struct registers
;
75 if(ptrace(PTRACE_GETREGS
, pid
, NULL
, ®isters
) == -1)
76 throw PTraceException(Misc::StreamAsString() << "Couldn't get register values: " << strerror(errno
));
79 #if AESALON_PLATFORM == AESALON_PLATFORM_x86_64
80 case ASM::Register::RAX
:
81 /*std::cout << "Value of RAX requested; RAX is " << registers.rax << ", ORIG_RAX is " << registers.orig_rax << std::endl;*/
83 case ASM::Register::RBX
:
85 case ASM::Register::RDI
:
87 case ASM::Register::RSI
:
89 case ASM::Register::RIP
:
91 case ASM::Register::RBP
:
93 case ASM::Register::RSP
:
97 throw PTraceException("Value of invalid register requested");
101 void Portal::set_register(ASM::Register which
, Word new_value
) {
102 struct user_regs_struct registers
;
103 if(ptrace(PTRACE_GETREGS
, pid
, NULL
, ®isters
) == -1)
104 throw PTraceException(Misc::StreamAsString() << "Couldn't get register values: " << strerror(errno
));
107 #if AESALON_PLATFORM == AESALON_PLATFORM_x86_64
108 case ASM::Register::RIP
:
109 registers
.rip
= new_value
;
113 throw PTraceException("Asked to set value of invalid register");
115 if(ptrace(PTRACE_SETREGS
, pid
, NULL
, ®isters
) == -1)
116 throw PTraceException(Misc::StreamAsString() << "Couldn't set register values: " << strerror(errno
));
119 Word
Portal::read_memory(Word address
) const {
120 std::cout
<< "Portal::read_memory() called . . ." << std::endl
;
121 std::cout
<< "\tReading address " << std::hex
<< address
<< std::endl
;
122 Word return_value
= ptrace(PTRACE_PEEKDATA
, pid
, address
, NULL
);
123 if(return_value
== Word(-1) && errno
!= 0)
124 throw PTraceException(Misc::StreamAsString() << "Couldn't read memory: " << strerror(errno
));
128 void Portal::write_memory(Word address
, Word value
) {
129 std::cout
<< "Portal::write_memory(address, Word) called . . ." << std::endl
;
130 if(ptrace(PTRACE_POKEDATA
, pid
, address
, value
) == -1)
131 throw PTraceException(Misc::StreamAsString() << "Couldn't write memory: " << strerror(errno
));
134 void Portal::write_memory(Word address
, Byte value
) {
135 std::cout
<< "Portal::write_memory(address, Byte) called . . ." << std::endl
;
136 std::cout
<< "\tWriting 0x" << std::hex
<< (int)value
<< " to " << address
<< std::endl
;
137 Word word_offset
= 0;
138 word_offset
= address
& 0x07;
139 Word current_value
= read_memory(address
- word_offset
);
141 current_value
&= ~(Word(0xff) << (word_offset
* CHAR_BIT
));
142 current_value
|= (Word(value
) << (word_offset
* CHAR_BIT
));
143 write_memory(address
- word_offset
, current_value
);
146 void Portal::attach() {
147 ptrace(PTRACE_ATTACH
, pid
, NULL
, NULL
);
150 void Portal::detach() {
151 ptrace(PTRACE_DETACH
, pid
, NULL
, NULL
);
154 std::size_t Portal::place_breakpoint(Word address
, Misc::SmartPointer
<BreakpointObserver
> observer
) {
155 std::cout
<< "Portal::place_breakpoint() called . . ." << std::endl
;
156 std::cout
<< "\tPlacing breakpoint at " << std::hex
<< address
<< std::dec
<< std::endl
;
157 Misc::SmartPointer
<Breakpoint
> bp
= get_breakpoint_by_address(address
);
159 Byte original
= read_memory(address
) & 0xff;
160 bp
= new Breakpoint(address
, original
);
162 write_memory(address
, bp
->get_breakpoint_character());
164 bp
->add_observer(observer
);
165 std::cout
<< "\tplaced breakpoint #" << bp
->get_id() << std::endl
;
169 void Portal::remove_breakpoint(Word address
) {
170 std::cout
<< "Portal::remove_breakpoint() called . . ." << std::endl
;
171 std::cout
<< "\tRemoving breakpoint at " << std::hex
<< address
<< std::dec
<< std::endl
;
173 Misc::SmartPointer
<Breakpoint
> bp
= get_breakpoint_by_address(address
);
175 std::cout
<< "\tAsked to remove non-existent breakpoint!" << std::endl
;
178 Byte original
= bp
->get_original();
179 write_memory(bp
->get_address(), original
);
180 for(breakpoint_list_t::iterator i
= breakpoint_list
.begin(); i
!= breakpoint_list
.end(); i
++) {
181 if((*i
)->get_id() == bp
->get_id()) {
182 breakpoint_list
.erase(i
);
188 Misc::SmartPointer
<Breakpoint
> Portal::get_breakpoint_by_id(std::size_t which
) const {
189 for(breakpoint_list_t::const_iterator i
= breakpoint_list
.begin(); i
!= breakpoint_list
.end(); i
++) {
190 if((*i
)->get_id() == which
) return *i
;
195 Misc::SmartPointer
<Breakpoint
> Portal::get_breakpoint_by_address(Word address
) const {
196 for(breakpoint_list_t::const_iterator i
= breakpoint_list
.begin(); i
!= breakpoint_list
.end(); i
++) {
197 if((*i
)->get_address() == address
) return *i
;
203 void Portal::handle_signal() {
205 int status
= wait_for_signal();
206 if(!WIFEXITED(status
) && WIFSTOPPED(status
)) {
207 siginfo_t signal_info
;
208 if(ptrace(PTRACE_GETSIGINFO
, pid
, NULL
, &signal_info
) == -1) {
209 throw PTraceException(Misc::StreamAsString() << "Failed to get signal information: " << strerror(errno
));
211 signal
= signal_info
.si_signo
;
215 std::cout
<< std::dec
;
217 std::cout
<< "Portal::handle_signal(): status: (" << status
<< "): ";
219 for(int mask
= 1 << 20; mask
> 0; mask
>>= 1) {
220 std::cout
<< ((status
& mask
)?"1":"0");
223 std::cout
<< ", signal: " << signal
<< std::endl
;
225 for(signal_observer_list_t::iterator i
= signal_observer_list
.begin(); i
!= signal_observer_list
.end(); i
++) {
226 if((*i
)->handle_signal(signal
, status
)) return;
228 Message::Message(Message::DEBUG_MESSAGE
, Misc::StreamAsString() << "Caught unknown signal " << signal
<< ", resuming execution with signal");
229 /* Default action: continue */
230 continue_execution(signal
);
233 void Portal::continue_execution(int signal
) {
234 std::cout
<< "Portal::continue_execution() called . . ." << std::endl
;
235 if(ptrace(PTRACE_CONT
, pid
, NULL
, NULL
) == -1)
236 throw PTraceException(Misc::StreamAsString() << "Couldn't continue program execution: " << strerror(errno
));
237 std::cout
<< "\tExecution continued." << std::endl
;
240 void Portal::single_step() {
241 std::cout
<< "Portal::single_step() called . . ." << std::endl
;
242 if(ptrace(PTRACE_SINGLESTEP
, pid
, NULL
, NULL
) == -1)
243 throw PTraceException(Misc::StreamAsString() << "Couldn't single-step program:" << strerror(errno
));
244 wait_for_signal(); /* ptrace(PTRACE_SINGLESTEP throws a SIGTRAP when it's done, so wait for it. */
245 std::cout
<< "\tSingle-step successful. " << std::endl
;
248 int Portal::wait_for_signal() {
249 std::cout
<< "Portal::wait_for_signal() called . . ." << std::endl
;
251 if(waitpid(pid
, &status
, 0) == -1) throw PTraceException(Misc::StreamAsString() << "Couldn't waitpid() on child: " << strerror(errno
));
252 std::cout
<< "\tGot signal." << std::endl
;
256 void Portal::handle_breakpoint() {
257 std::cout
<< "Portal::handle_breakpoint() called . . ." << std::endl
;
258 Word ip
= get_register(
259 #if AESALON_PLATFORM == AESALON_PLATFORM_x86_64
261 #elif AESALON_PLATFORM == AESALON_PLATFORM_x86
265 /* Subtract one from the IP, since the 0xcc SIGTRAP instruction was executed . . . */
267 Misc::SmartPointer
<Breakpoint
> breakpoint
= get_breakpoint_by_address(ip
);
268 std::cout
<< "\tIP: " << std::hex
<< ip
<< std::dec
<< std::endl
;
269 if(!breakpoint
.is_valid()) {
270 /*Message(Message::DEBUG_MESSAGE, "handle_breakpoint() called on non-breakpoint");*/
273 Message(Message::DEBUG_MESSAGE
, "handle_breakpoint() found a breakpoint . . .");
275 /* ip is currently ($rip - 1), to use gdb notation. In other words, back up one byte. */
276 set_register(ASM::Register::RIP
, ip
);
278 breakpoint
->notify();
280 /* Write out the original instruction . . . */
281 write_memory(breakpoint
->get_address(), breakpoint
->get_original());
282 /* Single-step over that instruction . . . */
284 /* Only re-write out the trap instruction if the breakpoint is still valid. */
285 if(breakpoint
->is_valid()) {
286 /* Re-write out the trap instruction . . . */
287 write_memory(breakpoint
->get_address(), breakpoint
->get_breakpoint_character());
290 remove_breakpoint(breakpoint
->get_address());
292 /* And then TrapObserver will call continue_execution(). */
295 Word
Portal::get_libc_offset() {
296 if(libc_offset
!= 0) return libc_offset
;
299 std::string map_file
;
300 map_file
= Misc::StreamAsString() << "/proc/" << pid
<< "/maps";
301 std::ifstream
map_stream(map_file
.c_str());
302 if(!map_stream
.is_open()) throw PTraceException(Misc::StreamAsString() << "Couldn't open " << map_file
<< ", perhaps permissions are screwy?");
304 /*std::cout << "Memory map: " << std::endl;*/
306 /* Example memory map excerpt (from bash):
307 00400000-004d5000 r-xp 00000000 08:06 71555 /bin/bash
308 006d4000-006de000 rw-p 000d4000 08:06 71555 /bin/bash
309 006de000-006e3000 rw-p 00000000 00:00 0
310 015ba000-01a44000 rw-p 00000000 00:00 0 [heap]
311 7f7698328000-7f769832a000 r-xp 00000000 08:06 218888 /usr/lib/gconv/ISO8859-1.so
312 7f769832a000-7f7698529000 ---p 00002000 08:06 218888 /usr/lib/gconv/ISO8859-1.so
313 7f7698529000-7f769852a000 r--p 00001000 08:06 218888 /usr/lib/gconv/ISO8859-1.so
314 7f769852a000-7f769852b000 rw-p 00002000 08:06 218888 /usr/lib/gconv/ISO8859-1.so
317 /* Get one line at a time, for EOF detection purposes. */
320 while(!map_stream
.eof() && map_stream
.getline(buffer
, 1024, '\n') && std::strlen(buffer
)) {
324 std::string mode
, device
, inode
, path
;
326 std::stringstream line_stream
;
327 line_stream
<< buffer
;
328 line_stream
>> std::hex
>> from
;
329 line_stream
>> spacer
;
330 line_stream
>> std::hex
>> to
;
332 line_stream
>> std::hex
>> offset
;
333 line_stream
>> device
;
334 line_stream
>> inode
;
336 /*std::cout << "\tLibrary path is \"" << path << "\"\n";
337 std::cout << "\tAddress range is: 0x" << std::hex << from << " to 0x" << to << " (size 0x" << to-from << ")" << std::endl;
338 std::cout << "\tMap mode is: \"" << mode << "\"\n";
339 std::cout << std::dec << std::endl;*/
340 if(Misc::String::begins_with(path
, "/lib/libc")) {
341 if(mode
== "r-xp") libc_offset
= from
;
350 } // namespace PTrace
351 } // namespace Monitor
352 } // namespace Aesalon