2284 lines
70 KiB
Perl
Executable File
2284 lines
70 KiB
Perl
Executable File
#!/usr/bin/perl
|
|
|
|
use strict;
|
|
|
|
#----------------------------------------------------------------------
|
|
# Globals
|
|
#----------------------------------------------------------------------
|
|
our $unimplemented_str = "UNIMPLEMENTED";
|
|
our $success_str = "OK";
|
|
our $swap = 1;
|
|
our $addr_size = 4;
|
|
our $thread_suffix_supported = 0;
|
|
our $max_bytes_per_line = 32;
|
|
our $addr_format = sprintf("0x%%%u.%ux", $addr_size*2, $addr_size*2);
|
|
our $pid_format = "%04.4x";
|
|
our $tid_format = "%04.4x";
|
|
our $reg8_href = { extract => \&get8, format => "0x%2.2x" };
|
|
our $reg16_href = { extract => \&get16, format => "0x%4.4x" };
|
|
our $reg32_href = { extract => \&get32, format => "0x%8.8x" };
|
|
our $reg64_href = { extract => \&get64, format => "0x%s" };
|
|
our $reg80_href = { extract => \&get80, format => "0x%s" };
|
|
our $reg128_href = { extract => \&get128, format => "0x%s" };
|
|
our $reg256_href = { extract => \&get256, format => "0x%s" };
|
|
our $float32_href = { extract => \&get32, format => "0x%8.8x" };
|
|
our $float64_href = { extract => \&get64, format => "0x%s" };
|
|
our $float96_href = { extract => \&get96, format => "0x%s" };
|
|
our $curr_cmd = undef;
|
|
our $curr_full_cmd = undef;
|
|
our %packet_times;
|
|
our $curr_time = 0.0;
|
|
our $last_time = 0.0;
|
|
our $base_time = 0.0;
|
|
our $packet_start_time = 0.0;
|
|
our $reg_cmd_reg;
|
|
our %reg_map = (
|
|
'i386-gdb' => [
|
|
{ name => 'eax', info => $reg32_href },
|
|
{ name => 'ecx', info => $reg32_href },
|
|
{ name => 'edx', info => $reg32_href },
|
|
{ name => 'ebx', info => $reg32_href },
|
|
{ name => 'esp', info => $reg32_href },
|
|
{ name => 'ebp', info => $reg32_href },
|
|
{ name => 'esi', info => $reg32_href },
|
|
{ name => 'edi', info => $reg32_href },
|
|
{ name => 'eip', info => $reg32_href },
|
|
{ name => 'eflags', info => $reg32_href },
|
|
{ name => 'cs', info => $reg32_href },
|
|
{ name => 'ss', info => $reg32_href },
|
|
{ name => 'ds', info => $reg32_href },
|
|
{ name => 'es', info => $reg32_href },
|
|
{ name => 'fs', info => $reg32_href },
|
|
{ name => 'gs', info => $reg32_href },
|
|
{ name => 'st0', info => $reg80_href },
|
|
{ name => 'st1', info => $reg80_href },
|
|
{ name => 'st2', info => $reg80_href },
|
|
{ name => 'st3', info => $reg80_href },
|
|
{ name => 'st4', info => $reg80_href },
|
|
{ name => 'st5', info => $reg80_href },
|
|
{ name => 'st6', info => $reg80_href },
|
|
{ name => 'st7', info => $reg80_href },
|
|
{ name => 'fctrl', info => $reg32_href },
|
|
{ name => 'fstat', info => $reg32_href },
|
|
{ name => 'ftag', info => $reg32_href },
|
|
{ name => 'fiseg', info => $reg32_href },
|
|
{ name => 'fioff', info => $reg32_href },
|
|
{ name => 'foseg', info => $reg32_href },
|
|
{ name => 'fooff', info => $reg32_href },
|
|
{ name => 'fop', info => $reg32_href },
|
|
{ name => 'xmm0', info => $reg128_href },
|
|
{ name => 'xmm1', info => $reg128_href },
|
|
{ name => 'xmm2', info => $reg128_href },
|
|
{ name => 'xmm3', info => $reg128_href },
|
|
{ name => 'xmm4', info => $reg128_href },
|
|
{ name => 'xmm5', info => $reg128_href },
|
|
{ name => 'xmm6', info => $reg128_href },
|
|
{ name => 'xmm7', info => $reg128_href },
|
|
{ name => 'mxcsr', info => $reg32_href },
|
|
{ name => 'mm0', info => $reg64_href },
|
|
{ name => 'mm1', info => $reg64_href },
|
|
{ name => 'mm2', info => $reg64_href },
|
|
{ name => 'mm3', info => $reg64_href },
|
|
{ name => 'mm4', info => $reg64_href },
|
|
{ name => 'mm5', info => $reg64_href },
|
|
{ name => 'mm6', info => $reg64_href },
|
|
{ name => 'mm7', info => $reg64_href },
|
|
],
|
|
|
|
'i386-lldb' => [
|
|
{ name => 'eax', info => $reg32_href },
|
|
{ name => 'ebx', info => $reg32_href },
|
|
{ name => 'ecx', info => $reg32_href },
|
|
{ name => 'edx', info => $reg32_href },
|
|
{ name => 'edi', info => $reg32_href },
|
|
{ name => 'esi', info => $reg32_href },
|
|
{ name => 'ebp', info => $reg32_href },
|
|
{ name => 'esp', info => $reg32_href },
|
|
{ name => 'ss', info => $reg32_href },
|
|
{ name => 'eflags', info => $reg32_href },
|
|
{ name => 'eip', info => $reg32_href },
|
|
{ name => 'cs', info => $reg32_href },
|
|
{ name => 'ds', info => $reg32_href },
|
|
{ name => 'es', info => $reg32_href },
|
|
{ name => 'fs', info => $reg32_href },
|
|
{ name => 'gs', info => $reg32_href },
|
|
{ name => 'fctrl', info => $reg16_href },
|
|
{ name => 'fstat', info => $reg16_href },
|
|
{ name => 'ftag', info => $reg8_href },
|
|
{ name => 'fop', info => $reg16_href },
|
|
{ name => 'fioff', info => $reg32_href },
|
|
{ name => 'fiseg', info => $reg16_href },
|
|
{ name => 'fooff', info => $reg32_href },
|
|
{ name => 'foseg', info => $reg16_href },
|
|
{ name => 'mxcsr', info => $reg32_href },
|
|
{ name => 'mxcsrmask', info => $reg32_href },
|
|
{ name => 'stmm0', info => $reg80_href },
|
|
{ name => 'stmm1', info => $reg80_href },
|
|
{ name => 'stmm2', info => $reg80_href },
|
|
{ name => 'stmm3', info => $reg80_href },
|
|
{ name => 'stmm4', info => $reg80_href },
|
|
{ name => 'stmm5', info => $reg80_href },
|
|
{ name => 'stmm6', info => $reg80_href },
|
|
{ name => 'stmm7', info => $reg80_href },
|
|
{ name => 'xmm0', info => $reg128_href },
|
|
{ name => 'xmm1', info => $reg128_href },
|
|
{ name => 'xmm2', info => $reg128_href },
|
|
{ name => 'xmm3', info => $reg128_href },
|
|
{ name => 'xmm4', info => $reg128_href },
|
|
{ name => 'xmm5', info => $reg128_href },
|
|
{ name => 'xmm6', info => $reg128_href },
|
|
{ name => 'xmm7', info => $reg128_href },
|
|
{ name => 'trapno', info => $reg32_href },
|
|
{ name => 'err', info => $reg32_href },
|
|
{ name => 'faultvaddr', info => $reg32_href },
|
|
],
|
|
|
|
'arm-gdb' => [
|
|
{ name => 'r0' , info => $reg32_href },
|
|
{ name => 'r1' , info => $reg32_href },
|
|
{ name => 'r2' , info => $reg32_href },
|
|
{ name => 'r3' , info => $reg32_href },
|
|
{ name => 'r4' , info => $reg32_href },
|
|
{ name => 'r5' , info => $reg32_href },
|
|
{ name => 'r6' , info => $reg32_href },
|
|
{ name => 'r7' , info => $reg32_href },
|
|
{ name => 'r8' , info => $reg32_href },
|
|
{ name => 'r9' , info => $reg32_href },
|
|
{ name => 'r10' , info => $reg32_href },
|
|
{ name => 'r11' , info => $reg32_href },
|
|
{ name => 'r12' , info => $reg32_href },
|
|
{ name => 'sp' , info => $reg32_href },
|
|
{ name => 'lr' , info => $reg32_href },
|
|
{ name => 'pc' , info => $reg32_href },
|
|
{ name => 'f0' , info => $float96_href },
|
|
{ name => 'f1' , info => $float96_href },
|
|
{ name => 'f2' , info => $float96_href },
|
|
{ name => 'f3' , info => $float96_href },
|
|
{ name => 'f4' , info => $float96_href },
|
|
{ name => 'f5' , info => $float96_href },
|
|
{ name => 'f6' , info => $float96_href },
|
|
{ name => 'f7' , info => $float96_href },
|
|
{ name => 'fps' , info => $reg32_href },
|
|
{ name => 'cpsr' , info => $reg32_href },
|
|
{ name => 's0' , info => $float32_href },
|
|
{ name => 's1' , info => $float32_href },
|
|
{ name => 's2' , info => $float32_href },
|
|
{ name => 's3' , info => $float32_href },
|
|
{ name => 's4' , info => $float32_href },
|
|
{ name => 's5' , info => $float32_href },
|
|
{ name => 's6' , info => $float32_href },
|
|
{ name => 's7' , info => $float32_href },
|
|
{ name => 's8' , info => $float32_href },
|
|
{ name => 's9' , info => $float32_href },
|
|
{ name => 's10' , info => $float32_href },
|
|
{ name => 's11' , info => $float32_href },
|
|
{ name => 's12' , info => $float32_href },
|
|
{ name => 's13' , info => $float32_href },
|
|
{ name => 's14' , info => $float32_href },
|
|
{ name => 's15' , info => $float32_href },
|
|
{ name => 's16' , info => $float32_href },
|
|
{ name => 's17' , info => $float32_href },
|
|
{ name => 's18' , info => $float32_href },
|
|
{ name => 's19' , info => $float32_href },
|
|
{ name => 's20' , info => $float32_href },
|
|
{ name => 's21' , info => $float32_href },
|
|
{ name => 's22' , info => $float32_href },
|
|
{ name => 's23' , info => $float32_href },
|
|
{ name => 's24' , info => $float32_href },
|
|
{ name => 's25' , info => $float32_href },
|
|
{ name => 's26' , info => $float32_href },
|
|
{ name => 's27' , info => $float32_href },
|
|
{ name => 's28' , info => $float32_href },
|
|
{ name => 's29' , info => $float32_href },
|
|
{ name => 's30' , info => $float32_href },
|
|
{ name => 's31' , info => $float32_href },
|
|
{ name => 'fpscr' , info => $reg32_href },
|
|
{ name => 'd16' , info => $float64_href },
|
|
{ name => 'd17' , info => $float64_href },
|
|
{ name => 'd18' , info => $float64_href },
|
|
{ name => 'd19' , info => $float64_href },
|
|
{ name => 'd20' , info => $float64_href },
|
|
{ name => 'd21' , info => $float64_href },
|
|
{ name => 'd22' , info => $float64_href },
|
|
{ name => 'd23' , info => $float64_href },
|
|
{ name => 'd24' , info => $float64_href },
|
|
{ name => 'd25' , info => $float64_href },
|
|
{ name => 'd26' , info => $float64_href },
|
|
{ name => 'd27' , info => $float64_href },
|
|
{ name => 'd28' , info => $float64_href },
|
|
{ name => 'd29' , info => $float64_href },
|
|
{ name => 'd30' , info => $float64_href },
|
|
{ name => 'd31' , info => $float64_href },
|
|
],
|
|
|
|
|
|
'arm-lldb' => [
|
|
{ name => 'r0' , info => $reg32_href },
|
|
{ name => 'r1' , info => $reg32_href },
|
|
{ name => 'r2' , info => $reg32_href },
|
|
{ name => 'r3' , info => $reg32_href },
|
|
{ name => 'r4' , info => $reg32_href },
|
|
{ name => 'r5' , info => $reg32_href },
|
|
{ name => 'r6' , info => $reg32_href },
|
|
{ name => 'r7' , info => $reg32_href },
|
|
{ name => 'r8' , info => $reg32_href },
|
|
{ name => 'r9' , info => $reg32_href },
|
|
{ name => 'r10' , info => $reg32_href },
|
|
{ name => 'r11' , info => $reg32_href },
|
|
{ name => 'r12' , info => $reg32_href },
|
|
{ name => 'sp' , info => $reg32_href },
|
|
{ name => 'lr' , info => $reg32_href },
|
|
{ name => 'pc' , info => $reg32_href },
|
|
{ name => 'cpsr' , info => $reg32_href },
|
|
{ name => 's0' , info => $float32_href },
|
|
{ name => 's1' , info => $float32_href },
|
|
{ name => 's2' , info => $float32_href },
|
|
{ name => 's3' , info => $float32_href },
|
|
{ name => 's4' , info => $float32_href },
|
|
{ name => 's5' , info => $float32_href },
|
|
{ name => 's6' , info => $float32_href },
|
|
{ name => 's7' , info => $float32_href },
|
|
{ name => 's8' , info => $float32_href },
|
|
{ name => 's9' , info => $float32_href },
|
|
{ name => 's10' , info => $float32_href },
|
|
{ name => 's11' , info => $float32_href },
|
|
{ name => 's12' , info => $float32_href },
|
|
{ name => 's13' , info => $float32_href },
|
|
{ name => 's14' , info => $float32_href },
|
|
{ name => 's15' , info => $float32_href },
|
|
{ name => 's16' , info => $float32_href },
|
|
{ name => 's17' , info => $float32_href },
|
|
{ name => 's18' , info => $float32_href },
|
|
{ name => 's19' , info => $float32_href },
|
|
{ name => 's20' , info => $float32_href },
|
|
{ name => 's21' , info => $float32_href },
|
|
{ name => 's22' , info => $float32_href },
|
|
{ name => 's23' , info => $float32_href },
|
|
{ name => 's24' , info => $float32_href },
|
|
{ name => 's25' , info => $float32_href },
|
|
{ name => 's26' , info => $float32_href },
|
|
{ name => 's27' , info => $float32_href },
|
|
{ name => 's28' , info => $float32_href },
|
|
{ name => 's29' , info => $float32_href },
|
|
{ name => 's30' , info => $float32_href },
|
|
{ name => 's31' , info => $float32_href },
|
|
{ name => 'd0' , info => $float64_href },
|
|
{ name => 'd1' , info => $float64_href },
|
|
{ name => 'd2' , info => $float64_href },
|
|
{ name => 'd3' , info => $float64_href },
|
|
{ name => 'd4' , info => $float64_href },
|
|
{ name => 'd5' , info => $float64_href },
|
|
{ name => 'd6' , info => $float64_href },
|
|
{ name => 'd7' , info => $float64_href },
|
|
{ name => 'd8' , info => $float64_href },
|
|
{ name => 'd9' , info => $float64_href },
|
|
{ name => 'd10' , info => $float64_href },
|
|
{ name => 'd11' , info => $float64_href },
|
|
{ name => 'd12' , info => $float64_href },
|
|
{ name => 'd13' , info => $float64_href },
|
|
{ name => 'd14' , info => $float64_href },
|
|
{ name => 'd15' , info => $float64_href },
|
|
{ name => 'd16' , info => $float64_href },
|
|
{ name => 'd17' , info => $float64_href },
|
|
{ name => 'd18' , info => $float64_href },
|
|
{ name => 'd19' , info => $float64_href },
|
|
{ name => 'd20' , info => $float64_href },
|
|
{ name => 'd21' , info => $float64_href },
|
|
{ name => 'd22' , info => $float64_href },
|
|
{ name => 'd23' , info => $float64_href },
|
|
{ name => 'd24' , info => $float64_href },
|
|
{ name => 'd25' , info => $float64_href },
|
|
{ name => 'd26' , info => $float64_href },
|
|
{ name => 'd27' , info => $float64_href },
|
|
{ name => 'd28' , info => $float64_href },
|
|
{ name => 'd29' , info => $float64_href },
|
|
{ name => 'd30' , info => $float64_href },
|
|
{ name => 'd31' , info => $float64_href },
|
|
{ name => 'fpscr' , info => $reg32_href },
|
|
{ name => 'exc' , info => $reg32_href },
|
|
{ name => 'fsr' , info => $reg32_href },
|
|
{ name => 'far' , info => $reg32_href },
|
|
],
|
|
|
|
'x86_64-gdb' => [
|
|
{ name => 'rax' , info => $reg64_href },
|
|
{ name => 'rbx' , info => $reg64_href },
|
|
{ name => 'rcx' , info => $reg64_href },
|
|
{ name => 'rdx' , info => $reg64_href },
|
|
{ name => 'rsi' , info => $reg64_href },
|
|
{ name => 'rdi' , info => $reg64_href },
|
|
{ name => 'rbp' , info => $reg64_href },
|
|
{ name => 'rsp' , info => $reg64_href },
|
|
{ name => 'r8' , info => $reg64_href },
|
|
{ name => 'r9' , info => $reg64_href },
|
|
{ name => 'r10' , info => $reg64_href },
|
|
{ name => 'r11' , info => $reg64_href },
|
|
{ name => 'r12' , info => $reg64_href },
|
|
{ name => 'r13' , info => $reg64_href },
|
|
{ name => 'r14' , info => $reg64_href },
|
|
{ name => 'r15' , info => $reg64_href },
|
|
{ name => 'rip' , info => $reg64_href },
|
|
{ name => 'eflags' , info => $reg32_href },
|
|
{ name => 'cs' , info => $reg32_href },
|
|
{ name => 'ss' , info => $reg32_href },
|
|
{ name => 'ds' , info => $reg32_href },
|
|
{ name => 'es' , info => $reg32_href },
|
|
{ name => 'fs' , info => $reg32_href },
|
|
{ name => 'gs' , info => $reg32_href },
|
|
{ name => 'stmm0' , info => $reg80_href },
|
|
{ name => 'stmm1' , info => $reg80_href },
|
|
{ name => 'stmm2' , info => $reg80_href },
|
|
{ name => 'stmm3' , info => $reg80_href },
|
|
{ name => 'stmm4' , info => $reg80_href },
|
|
{ name => 'stmm5' , info => $reg80_href },
|
|
{ name => 'stmm6' , info => $reg80_href },
|
|
{ name => 'stmm7' , info => $reg80_href },
|
|
{ name => 'fctrl' , info => $reg32_href },
|
|
{ name => 'fstat' , info => $reg32_href },
|
|
{ name => 'ftag' , info => $reg32_href },
|
|
{ name => 'fiseg' , info => $reg32_href },
|
|
{ name => 'fioff' , info => $reg32_href },
|
|
{ name => 'foseg' , info => $reg32_href },
|
|
{ name => 'fooff' , info => $reg32_href },
|
|
{ name => 'fop' , info => $reg32_href },
|
|
{ name => 'xmm0' , info => $reg128_href },
|
|
{ name => 'xmm1' , info => $reg128_href },
|
|
{ name => 'xmm2' , info => $reg128_href },
|
|
{ name => 'xmm3' , info => $reg128_href },
|
|
{ name => 'xmm4' , info => $reg128_href },
|
|
{ name => 'xmm5' , info => $reg128_href },
|
|
{ name => 'xmm6' , info => $reg128_href },
|
|
{ name => 'xmm7' , info => $reg128_href },
|
|
{ name => 'xmm8' , info => $reg128_href },
|
|
{ name => 'xmm9' , info => $reg128_href },
|
|
{ name => 'xmm10' , info => $reg128_href },
|
|
{ name => 'xmm11' , info => $reg128_href },
|
|
{ name => 'xmm12' , info => $reg128_href },
|
|
{ name => 'xmm13' , info => $reg128_href },
|
|
{ name => 'xmm14' , info => $reg128_href },
|
|
{ name => 'xmm15' , info => $reg128_href },
|
|
{ name => 'mxcsr' , info => $reg32_href },
|
|
],
|
|
|
|
'x86_64-lldb' => [
|
|
{ name => 'rax' , info => $reg64_href },
|
|
{ name => 'rbx' , info => $reg64_href },
|
|
{ name => 'rcx' , info => $reg64_href },
|
|
{ name => 'rdx' , info => $reg64_href },
|
|
{ name => 'rdi' , info => $reg64_href },
|
|
{ name => 'rsi' , info => $reg64_href },
|
|
{ name => 'rbp' , info => $reg64_href },
|
|
{ name => 'rsp' , info => $reg64_href },
|
|
{ name => 'r8 ' , info => $reg64_href },
|
|
{ name => 'r9 ' , info => $reg64_href },
|
|
{ name => 'r10' , info => $reg64_href },
|
|
{ name => 'r11' , info => $reg64_href },
|
|
{ name => 'r12' , info => $reg64_href },
|
|
{ name => 'r13' , info => $reg64_href },
|
|
{ name => 'r14' , info => $reg64_href },
|
|
{ name => 'r15' , info => $reg64_href },
|
|
{ name => 'rip' , info => $reg64_href },
|
|
{ name => 'rflags' , info => $reg64_href },
|
|
{ name => 'cs' , info => $reg64_href },
|
|
{ name => 'fs' , info => $reg64_href },
|
|
{ name => 'gs' , info => $reg64_href },
|
|
{ name => 'fctrl' , info => $reg16_href },
|
|
{ name => 'fstat' , info => $reg16_href },
|
|
{ name => 'ftag' , info => $reg8_href },
|
|
{ name => 'fop' , info => $reg16_href },
|
|
{ name => 'fioff' , info => $reg32_href },
|
|
{ name => 'fiseg' , info => $reg16_href },
|
|
{ name => 'fooff' , info => $reg32_href },
|
|
{ name => 'foseg' , info => $reg16_href },
|
|
{ name => 'mxcsr' , info => $reg32_href },
|
|
{ name => 'mxcsrmask' , info => $reg32_href },
|
|
{ name => 'stmm0' , info => $reg80_href },
|
|
{ name => 'stmm1' , info => $reg80_href },
|
|
{ name => 'stmm2' , info => $reg80_href },
|
|
{ name => 'stmm3' , info => $reg80_href },
|
|
{ name => 'stmm4' , info => $reg80_href },
|
|
{ name => 'stmm5' , info => $reg80_href },
|
|
{ name => 'stmm6' , info => $reg80_href },
|
|
{ name => 'stmm7' , info => $reg80_href },
|
|
{ name => 'xmm0' , info => $reg128_href },
|
|
{ name => 'xmm1' , info => $reg128_href },
|
|
{ name => 'xmm2' , info => $reg128_href },
|
|
{ name => 'xmm3' , info => $reg128_href },
|
|
{ name => 'xmm4' , info => $reg128_href },
|
|
{ name => 'xmm5' , info => $reg128_href },
|
|
{ name => 'xmm6' , info => $reg128_href },
|
|
{ name => 'xmm7' , info => $reg128_href },
|
|
{ name => 'xmm8' , info => $reg128_href },
|
|
{ name => 'xmm9' , info => $reg128_href },
|
|
{ name => 'xmm10' , info => $reg128_href },
|
|
{ name => 'xmm11' , info => $reg128_href },
|
|
{ name => 'xmm12' , info => $reg128_href },
|
|
{ name => 'xmm13' , info => $reg128_href },
|
|
{ name => 'xmm14' , info => $reg128_href },
|
|
{ name => 'xmm15' , info => $reg128_href },
|
|
{ name => 'trapno' , info => $reg32_href },
|
|
{ name => 'err' , info => $reg32_href },
|
|
{ name => 'faultvaddr' , info => $reg64_href },
|
|
]
|
|
);
|
|
|
|
our $max_register_name_len = 0;
|
|
calculate_max_register_name_length();
|
|
our @point_types = ( "software_bp", "hardware_bp", "write_wp", "read_wp", "access_wp" );
|
|
our $opt_v = 0; # verbose
|
|
our $opt_g = 0; # debug
|
|
our $opt_q = 0; # quiet
|
|
our $opt_r = undef;
|
|
use Getopt::Std;
|
|
getopts('gvqr:');
|
|
|
|
our $registers_aref = undef;
|
|
|
|
if (length($opt_r))
|
|
{
|
|
if (exists $reg_map{$opt_r})
|
|
{
|
|
$registers_aref = $reg_map{$opt_r};
|
|
}
|
|
else
|
|
{
|
|
die "Can't get registers group for '$opt_r'\n";
|
|
}
|
|
}
|
|
|
|
sub extract_key_value_pairs
|
|
{
|
|
my $kv_href = {};
|
|
my $arrayref = shift;
|
|
my $str = join('',@$arrayref);
|
|
my @kv_strs = split(/;/, $str);
|
|
foreach my $kv_str (@kv_strs)
|
|
{
|
|
my ($key, $value) = split(/:/, $kv_str);
|
|
$kv_href->{$key} = $value;
|
|
}
|
|
return $kv_href;
|
|
}
|
|
|
|
sub get_thread_from_thread_suffix
|
|
{
|
|
if ($thread_suffix_supported)
|
|
{
|
|
my $arrayref = shift;
|
|
# Skip leading semi-colon if needed
|
|
$$arrayref[0] == ';' and shift @$arrayref;
|
|
my $thread_href = extract_key_value_pairs ($arrayref);
|
|
if (exists $thread_href->{thread})
|
|
{
|
|
return $thread_href->{thread};
|
|
}
|
|
}
|
|
return undef;
|
|
}
|
|
|
|
sub calculate_max_register_name_length
|
|
{
|
|
$max_register_name_len = 7;
|
|
foreach my $reg_href (@$registers_aref)
|
|
{
|
|
my $name_len = length($reg_href->{name});
|
|
if ($max_register_name_len < $name_len)
|
|
{
|
|
$max_register_name_len = $name_len;
|
|
}
|
|
}
|
|
}
|
|
#----------------------------------------------------------------------
|
|
# Hash that maps command characters to the appropriate functions using
|
|
# the command character as the key and the value being a reference to
|
|
# the dump function for dumping the command itself.
|
|
#----------------------------------------------------------------------
|
|
our %cmd_callbacks =
|
|
(
|
|
'?' => \&dump_last_signal_cmd,
|
|
'H' => \&dump_set_thread_cmd,
|
|
'T' => \&dump_thread_is_alive_cmd,
|
|
'q' => \&dump_general_query_cmd,
|
|
'Q' => \&dump_general_set_cmd,
|
|
'g' => \&dump_read_regs_cmd,
|
|
'G' => \&dump_write_regs_cmd,
|
|
'p' => \&dump_read_single_register_cmd,
|
|
'P' => \&dump_write_single_register_cmd,
|
|
'm' => \&dump_read_mem_cmd,
|
|
'M' => \&dump_write_mem_cmd,
|
|
'X' => \&dump_write_mem_binary_cmd,
|
|
'Z' => \&dump_bp_wp_command,
|
|
'z' => \&dump_bp_wp_command,
|
|
'k' => \&dump_kill_cmd,
|
|
'A' => \&dump_A_command,
|
|
'c' => \&dump_continue_cmd,
|
|
's' => \&dump_continue_cmd,
|
|
'C' => \&dump_continue_with_signal_cmd,
|
|
'S' => \&dump_continue_with_signal_cmd,
|
|
'_M' => \&dump_allocate_memory_cmd,
|
|
'_m' => \&dump_deallocate_memory_cmd,
|
|
# extended commands
|
|
'v' => \&dump_extended_cmd
|
|
);
|
|
|
|
#----------------------------------------------------------------------
|
|
# Hash that maps command characters to the appropriate functions using
|
|
# the command character as the key and the value being a reference to
|
|
# the dump function for the response to the command.
|
|
#----------------------------------------------------------------------
|
|
our %rsp_callbacks =
|
|
(
|
|
'c' => \&dump_stop_reply_packet,
|
|
's' => \&dump_stop_reply_packet,
|
|
'C' => \&dump_stop_reply_packet,
|
|
'?' => \&dump_stop_reply_packet,
|
|
'T' => \&dump_thread_is_alive_rsp,
|
|
'H' => \&dump_set_thread_rsp,
|
|
'q' => \&dump_general_query_rsp,
|
|
'g' => \&dump_read_regs_rsp,
|
|
'p' => \&dump_read_single_register_rsp,
|
|
'm' => \&dump_read_mem_rsp,
|
|
'_M' => \&dump_allocate_memory_rsp,
|
|
|
|
# extended commands
|
|
'v' => \&dump_extended_rsp,
|
|
);
|
|
|
|
|
|
sub dump_register_value
|
|
{
|
|
my $indent = shift;
|
|
my $arrayref = shift;
|
|
my $reg_num = shift;
|
|
|
|
if ($reg_num >= @$registers_aref)
|
|
{
|
|
printf("\tinvalid register index %d\n", $reg_num);
|
|
return;
|
|
}
|
|
|
|
my $reg_href = $$registers_aref[$reg_num];
|
|
my $reg_name = $reg_href->{name};
|
|
if ($$arrayref[0] eq '#')
|
|
{
|
|
printf("\t%*s: error: EOS reached when trying to read register %d\n", $max_register_name_len, $reg_name, $reg_num);
|
|
}
|
|
|
|
my $reg_info = $reg_href->{info};
|
|
my $reg_extract = $reg_info->{extract};
|
|
my $reg_format = $reg_info->{format};
|
|
my $reg_val = &$reg_extract($arrayref);
|
|
if ($indent) {
|
|
printf("\t%*s = $reg_format", $max_register_name_len, $reg_name, $reg_val);
|
|
} else {
|
|
printf("%s = $reg_format", $reg_name, $reg_val);
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Extract the command into an array of ASCII char strings for easy
|
|
# processing
|
|
#----------------------------------------------------------------------
|
|
sub extract_command
|
|
{
|
|
my $cmd_str = shift;
|
|
my @cmd_chars = split(/ */, $cmd_str);
|
|
if ($cmd_chars[0] ne '$')
|
|
{
|
|
# only set the current command if it isn't a reply
|
|
$curr_cmd = $cmd_chars[0];
|
|
}
|
|
return @cmd_chars;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Strip the 3 checksum array entries after we don't need them anymore
|
|
#----------------------------------------------------------------------
|
|
sub strip_checksum
|
|
{
|
|
my $arrayref = shift;
|
|
splice(@$arrayref, -3);
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Dump all strings in array by joining them together with no space
|
|
# between them
|
|
#----------------------------------------------------------------------
|
|
sub dump_chars
|
|
{
|
|
print join('',@_);
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Check if the response is an error 'EXX'
|
|
#----------------------------------------------------------------------
|
|
sub is_error_response
|
|
{
|
|
if ($_[0] eq 'E')
|
|
{
|
|
shift;
|
|
print "ERROR = " . join('',@_) . "\n";
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'H' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_set_thread_cmd
|
|
{
|
|
my $cmd = shift;
|
|
my $mod = shift;
|
|
print "set_thread ( $mod, " . join('',@_) . " )\n";
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'T' command
|
|
#----------------------------------------------------------------------
|
|
our $T_cmd_tid = -1;
|
|
sub dump_thread_is_alive_cmd
|
|
{
|
|
my $cmd = shift;
|
|
$T_cmd_tid = get_hex(\@_);
|
|
printf("thread_is_alive ( $tid_format )\n", $T_cmd_tid);
|
|
}
|
|
|
|
sub dump_thread_is_alive_rsp
|
|
{
|
|
my $rsp = join('',@_);
|
|
|
|
printf("thread_is_alive ( $tid_format ) =>", $T_cmd_tid);
|
|
if ($rsp eq 'OK')
|
|
{
|
|
print " alive.\n";
|
|
}
|
|
else
|
|
{
|
|
print " dead.\n";
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'H' response
|
|
#----------------------------------------------------------------------
|
|
sub dump_set_thread_rsp
|
|
{
|
|
if (!is_error_response(@_))
|
|
{
|
|
print join('',@_) . "\n";
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'q' command
|
|
#----------------------------------------------------------------------
|
|
our $gen_query_cmd;
|
|
our $qRegisterInfo_reg_num = -1;
|
|
sub dump_general_query_cmd
|
|
{
|
|
$gen_query_cmd = join('',@_);
|
|
if ($gen_query_cmd eq 'qC')
|
|
{
|
|
print 'get_current_pid ()';
|
|
}
|
|
elsif ($gen_query_cmd eq 'qfThreadInfo')
|
|
{
|
|
print 'get_first_active_threads ()';
|
|
}
|
|
elsif ($gen_query_cmd eq 'qsThreadInfo')
|
|
{
|
|
print 'get_subsequent_active_threads ()';
|
|
}
|
|
elsif (index($gen_query_cmd, 'qThreadExtraInfo') == 0)
|
|
{
|
|
# qThreadExtraInfo,id
|
|
print 'get_thread_extra_info ()';
|
|
}
|
|
elsif (index($gen_query_cmd, 'qThreadStopInfo') == 0)
|
|
{
|
|
# qThreadStopInfoXXXX
|
|
@_ = splice(@_, length('qThreadStopInfo'));
|
|
my $tid = get_addr(\@_);
|
|
printf('get_thread_stop_info ( thread = 0x%4.4x )', $tid);
|
|
}
|
|
elsif (index($gen_query_cmd, 'qSymbol:') == 0)
|
|
{
|
|
# qCRC:addr,length
|
|
print 'gdb_ready_to_serve_symbol_lookups ()';
|
|
}
|
|
elsif (index($gen_query_cmd, 'qCRC:') == 0)
|
|
{
|
|
# qCRC:addr,length
|
|
@_ = splice(@_, length('qCRC:'));
|
|
my $address = get_addr(\@_);
|
|
shift @_;
|
|
my $length = join('', @_);
|
|
printf("compute_crc (addr = $addr_format, length = $length)", $address);
|
|
}
|
|
elsif (index($gen_query_cmd, 'qGetTLSAddr:') == 0)
|
|
{
|
|
# qGetTLSAddr:thread-id,offset,lm
|
|
@_ = splice(@_, length('qGetTLSAddr:'));
|
|
my ($tid, $offset, $lm) = split (/,/, join('', @_));
|
|
print "get_thread_local_storage_addr (thread-id = $tid, offset = $offset, lm = $lm)";
|
|
}
|
|
elsif ($gen_query_cmd eq 'qOffsets')
|
|
{
|
|
print 'get_section_offsets ()';
|
|
}
|
|
elsif (index($gen_query_cmd, 'qRegisterInfo') == 0)
|
|
{
|
|
@_ = splice(@_, length('qRegisterInfo'));
|
|
$qRegisterInfo_reg_num = get_hex(\@_);
|
|
|
|
printf "get_dynamic_register_info ($qRegisterInfo_reg_num)";
|
|
}
|
|
else
|
|
{
|
|
print $gen_query_cmd;
|
|
}
|
|
print "\n";
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'q' response
|
|
#----------------------------------------------------------------------
|
|
sub dump_general_query_rsp
|
|
{
|
|
my $gen_query_rsp = join('',@_);
|
|
my $gen_query_rsp_len = length ($gen_query_rsp);
|
|
if ($gen_query_cmd eq 'qC' and index($gen_query_rsp, 'QC') == 0)
|
|
{
|
|
shift @_; shift @_;
|
|
my $pid = get_hex(\@_);
|
|
printf("pid = $pid_format\n", $pid);
|
|
return;
|
|
}
|
|
elsif (index($gen_query_cmd, 'qRegisterInfo') == 0)
|
|
{
|
|
if ($gen_query_rsp_len == 0)
|
|
{
|
|
print "$unimplemented_str\n";
|
|
}
|
|
else
|
|
{
|
|
if (index($gen_query_rsp, 'name') == 0)
|
|
{
|
|
$qRegisterInfo_reg_num == 0 and $registers_aref = [];
|
|
|
|
my @name_and_values = split (/;/, $gen_query_rsp);
|
|
|
|
my $reg_name = undef;
|
|
my $byte_size = 0;
|
|
my $pseudo = 0;
|
|
foreach (@name_and_values)
|
|
{
|
|
my ($name, $value) = split /:/;
|
|
if ($name eq "name") { $reg_name = $value; }
|
|
elsif ($name eq "bitsize") { $byte_size = $value / 8; }
|
|
elsif ($name eq "container-regs") { $pseudo = 1; }
|
|
}
|
|
if (defined $reg_name and $byte_size > 0)
|
|
{
|
|
if ($byte_size == 4) {push @$registers_aref, { name => $reg_name, info => $reg32_href , pseudo => $pseudo };}
|
|
elsif ($byte_size == 8) {push @$registers_aref, { name => $reg_name, info => $reg64_href , pseudo => $pseudo };}
|
|
elsif ($byte_size == 1) {push @$registers_aref, { name => $reg_name, info => $reg8_href , pseudo => $pseudo };}
|
|
elsif ($byte_size == 2) {push @$registers_aref, { name => $reg_name, info => $reg16_href , pseudo => $pseudo };}
|
|
elsif ($byte_size == 10) {push @$registers_aref, { name => $reg_name, info => $reg80_href , pseudo => $pseudo };}
|
|
elsif ($byte_size == 12) {push @$registers_aref, { name => $reg_name, info => $float96_href , pseudo => $pseudo };}
|
|
elsif ($byte_size == 16) {push @$registers_aref, { name => $reg_name, info => $reg128_href , pseudo => $pseudo };}
|
|
elsif ($byte_size == 32) {push @$registers_aref, { name => $reg_name, info => $reg256_href , pseudo => $pseudo };}
|
|
}
|
|
}
|
|
elsif ($gen_query_rsp_len == 3 and index($gen_query_rsp, 'E') == 0)
|
|
{
|
|
calculate_max_register_name_length();
|
|
}
|
|
}
|
|
}
|
|
elsif ($gen_query_cmd =~ 'qThreadStopInfo')
|
|
{
|
|
dump_stop_reply_packet (@_);
|
|
}
|
|
if (dump_standard_response(\@_))
|
|
{
|
|
# Do nothing...
|
|
}
|
|
else
|
|
{
|
|
print join('',@_) . "\n";
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'Q' command
|
|
#----------------------------------------------------------------------
|
|
our $gen_set_cmd;
|
|
sub dump_general_set_cmd
|
|
{
|
|
$gen_query_cmd = join('',@_);
|
|
if ($gen_query_cmd eq 'QStartNoAckMode')
|
|
{
|
|
print "StartNoAckMode ()"
|
|
}
|
|
elsif ($gen_query_cmd eq 'QThreadSuffixSupported')
|
|
{
|
|
$thread_suffix_supported = 1;
|
|
print "ThreadSuffixSupported ()"
|
|
}
|
|
elsif (index($gen_query_cmd, 'QSetMaxPayloadSize:') == 0)
|
|
{
|
|
@_ = splice(@_, length('QSetMaxPayloadSize:'));
|
|
my $max_payload_size = get_hex(\@_);
|
|
# QSetMaxPayloadSize:XXXX where XXXX is a hex length of the max
|
|
# packet payload size supported by gdb
|
|
printf("SetMaxPayloadSize ( 0x%x (%u))", $max_payload_size, $max_payload_size);
|
|
}
|
|
elsif (index ($gen_query_cmd, 'QSetSTDIN:') == 0)
|
|
{
|
|
@_ = splice(@_, length('QSetSTDIN:'));
|
|
printf ("SetSTDIN (path ='%s')\n", get_hex_string (\@_));
|
|
}
|
|
elsif (index ($gen_query_cmd, 'QSetSTDOUT:') == 0)
|
|
{
|
|
@_ = splice(@_, length('QSetSTDOUT:'));
|
|
printf ("SetSTDOUT (path ='%s')\n", get_hex_string (\@_));
|
|
}
|
|
elsif (index ($gen_query_cmd, 'QSetSTDERR:') == 0)
|
|
{
|
|
@_ = splice(@_, length('QSetSTDERR:'));
|
|
printf ("SetSTDERR (path ='%s')\n", get_hex_string (\@_));
|
|
}
|
|
else
|
|
{
|
|
print $gen_query_cmd;
|
|
}
|
|
print "\n";
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'k' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_kill_cmd
|
|
{
|
|
my $cmd = shift;
|
|
print "kill (" . join('',@_) . ")\n";
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'g' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_read_regs_cmd
|
|
{
|
|
my $cmd = shift;
|
|
print "read_registers ()\n";
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'G' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_write_regs_cmd
|
|
{
|
|
print "write_registers:\n";
|
|
my $cmd = shift;
|
|
foreach my $reg_href (@$registers_aref)
|
|
{
|
|
last if ($_[0] eq '#');
|
|
if ($reg_href->{pseudo} == 0)
|
|
{
|
|
my $reg_info_href = $reg_href->{info};
|
|
my $reg_name = $reg_href->{name};
|
|
my $reg_extract = $reg_info_href->{extract};
|
|
my $reg_format = $reg_info_href->{format};
|
|
my $reg_val = &$reg_extract(\@_);
|
|
printf("\t%*s = $reg_format\n", $max_register_name_len, $reg_name, $reg_val);
|
|
}
|
|
}
|
|
}
|
|
|
|
sub dump_read_regs_rsp
|
|
{
|
|
print "read_registers () =>\n";
|
|
if (!is_error_response(@_))
|
|
{
|
|
# print join('',@_) . "\n";
|
|
foreach my $reg_href (@$registers_aref)
|
|
{
|
|
last if ($_[0] eq '#');
|
|
if ($reg_href->{pseudo} == 0)
|
|
{
|
|
my $reg_info_href = $reg_href->{info};
|
|
my $reg_name = $reg_href->{name};
|
|
my $reg_extract = $reg_info_href->{extract};
|
|
my $reg_format = $reg_info_href->{format};
|
|
my $reg_val = &$reg_extract(\@_);
|
|
printf("\t%*s = $reg_format\n", $max_register_name_len, $reg_name, $reg_val);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
sub dump_read_single_register_rsp
|
|
{
|
|
dump_register_value(0, \@_, $reg_cmd_reg);
|
|
print "\n";
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# '_M' - allocate memory command (LLDB extension)
|
|
#
|
|
# Command: '_M'
|
|
# Arg1: Hex byte size as big endian hex string
|
|
# Separator: ','
|
|
# Arg2: permissions as string that must be a string that contains any
|
|
# combination of 'r' (readable) 'w' (writable) or 'x' (executable)
|
|
#
|
|
# Returns: The address that was allocated as a big endian hex string
|
|
# on success, else an error "EXX" where XX are hex bytes
|
|
# that indicate an error code.
|
|
#
|
|
# Examples:
|
|
# _M10,rw # allocate 16 bytes with read + write permissions
|
|
# _M100,rx # allocate 256 bytes with read + execute permissions
|
|
#----------------------------------------------------------------------
|
|
sub dump_allocate_memory_cmd
|
|
{
|
|
shift; shift; # shift off the '_' and the 'M'
|
|
my $byte_size = get_addr(\@_);
|
|
shift; # Skip ','
|
|
printf("allocate_memory ( byte_size = %u (0x%x), permissions = %s)\n", $byte_size, $byte_size, join('',@_));
|
|
}
|
|
|
|
sub dump_allocate_memory_rsp
|
|
{
|
|
if (@_ == 3 and $_[0] == 'E')
|
|
{
|
|
printf("allocated memory addr = ERROR (%s))\n", join('',@_));
|
|
}
|
|
else
|
|
{
|
|
printf("allocated memory addr = 0x%s\n", join('',@_));
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# '_m' - deallocate memory command (LLDB extension)
|
|
#
|
|
# Command: '_m'
|
|
# Arg1: Hex address as big endian hex string
|
|
#
|
|
# Returns: "OK" on success "EXX" on error
|
|
#
|
|
# Examples:
|
|
# _m201000 # Free previously allocated memory at address 0x201000
|
|
#----------------------------------------------------------------------
|
|
sub dump_deallocate_memory_cmd
|
|
{
|
|
shift; shift; # shift off the '_' and the 'm'
|
|
printf("deallocate_memory ( addr = 0x%s)\n", join('',@_));
|
|
}
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'p' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_read_single_register_cmd
|
|
{
|
|
my $cmd = shift;
|
|
$reg_cmd_reg = get_hex(\@_);
|
|
my $thread = get_thread_from_thread_suffix (\@_);
|
|
my $reg_href = $$registers_aref[$reg_cmd_reg];
|
|
|
|
if (defined $thread)
|
|
{
|
|
print "read_register ( reg = \"$reg_href->{name}\", thread = $thread )\n";
|
|
}
|
|
else
|
|
{
|
|
print "read_register ( reg = \"$reg_href->{name}\" )\n";
|
|
}
|
|
}
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'P' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_write_single_register_cmd
|
|
{
|
|
my $cmd = shift;
|
|
my $reg_num = get_hex(\@_);
|
|
shift (@_); # Discard the '='
|
|
|
|
print "write_register ( ";
|
|
dump_register_value(0, \@_, $reg_num);
|
|
my $thread = get_thread_from_thread_suffix (\@_);
|
|
if (defined $thread)
|
|
{
|
|
print ", thread = $thread";
|
|
}
|
|
print " )\n";
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'm' command
|
|
#----------------------------------------------------------------------
|
|
our $read_mem_address = 0;
|
|
sub dump_read_mem_cmd
|
|
{
|
|
my $cmd = shift;
|
|
$read_mem_address = get_addr(\@_);
|
|
shift; # Skip ','
|
|
printf("read_mem ( $addr_format, %s )\n", $read_mem_address, join('',@_));
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'm' response
|
|
#----------------------------------------------------------------------
|
|
sub dump_read_mem_rsp
|
|
{
|
|
# If the memory read was 2 or 4 bytes, print it out in native format
|
|
# instead of just as bytes.
|
|
my $num_nibbles = @_;
|
|
if ($num_nibbles == 2)
|
|
{
|
|
printf(" 0x%2.2x", get8(\@_));
|
|
}
|
|
elsif ($num_nibbles == 4)
|
|
{
|
|
printf(" 0x%4.4x", get16(\@_));
|
|
}
|
|
elsif ($num_nibbles == 8)
|
|
{
|
|
printf(" 0x%8.8x", get32(\@_));
|
|
}
|
|
elsif ($num_nibbles == 16)
|
|
{
|
|
printf(" 0x%s", get64(\@_));
|
|
}
|
|
else
|
|
{
|
|
my $curr_address = $read_mem_address;
|
|
my $nibble;
|
|
my $nibble_offset = 0;
|
|
my $max_nibbles_per_line = 2 * $max_bytes_per_line;
|
|
foreach $nibble (@_)
|
|
{
|
|
if (($nibble_offset % $max_nibbles_per_line) == 0)
|
|
{
|
|
($nibble_offset > 0) and print "\n ";
|
|
printf("$addr_format: ", $curr_address + $nibble_offset/2);
|
|
}
|
|
(($nibble_offset % 2) == 0) and print ' ';
|
|
print $nibble;
|
|
$nibble_offset++;
|
|
}
|
|
}
|
|
print "\n";
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'c' or 's' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_continue_cmd
|
|
{
|
|
my $cmd = shift;
|
|
my $cmd_str;
|
|
$cmd eq 'c' and $cmd_str = 'continue';
|
|
$cmd eq 's' and $cmd_str = 'step';
|
|
my $address = -1;
|
|
if (@_)
|
|
{
|
|
my $address = get_addr(\@_);
|
|
printf("%s ($addr_format)\n", $cmd_str, $address);
|
|
}
|
|
else
|
|
{
|
|
printf("%s ()\n", $cmd_str);
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'Css' continue (C) with signal (ss where 'ss' is two hex digits)
|
|
# 'Sss' step (S) with signal (ss where 'ss' is two hex digits)
|
|
#----------------------------------------------------------------------
|
|
sub dump_continue_with_signal_cmd
|
|
{
|
|
my $cmd = shift;
|
|
my $address = -1;
|
|
my $cmd_str;
|
|
$cmd eq 'c' and $cmd_str = 'continue';
|
|
$cmd eq 's' and $cmd_str = 'step';
|
|
my $signal = get_hex(\@_);
|
|
if (@_)
|
|
{
|
|
my $address = 0;
|
|
if (@_ && $_[0] == ';')
|
|
{
|
|
shift;
|
|
$address = get_addr(\@_);
|
|
}
|
|
}
|
|
|
|
if ($address != -1)
|
|
{
|
|
printf("%s_with_signal (signal = 0x%2.2x, address = $addr_format)\n", $cmd_str, $signal, $address);
|
|
}
|
|
else
|
|
{
|
|
printf("%s_with_signal (signal = 0x%2.2x)\n", $cmd_str, $signal);
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'A' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_A_command
|
|
{
|
|
my $cmd = get_expected_char(\@_, 'A') or print "error: incorrect command letter for argument packet, expected 'A'\n";
|
|
printf("set_program_arguments (\n");
|
|
do
|
|
{
|
|
my $arg_len = get_uint(\@_);
|
|
get_expected_char(\@_, ',') or die "error: missing comma after argument length...?\n";
|
|
my $arg_idx = get_uint(\@_);
|
|
get_expected_char(\@_, ',') or die "error: missing comma after argument number...?\n";
|
|
|
|
my $arg = '';
|
|
my $num_hex8_bytes = $arg_len/2;
|
|
for (1 .. $num_hex8_bytes)
|
|
{
|
|
$arg .= sprintf("%c", get8(\@_))
|
|
}
|
|
printf(" <%3u> argv[%u] = '%s'\n", $arg_len, $arg_idx, $arg);
|
|
if (@_ > 0)
|
|
{
|
|
get_expected_char(\@_, ',') or die "error: missing comma after argument argument ASCII hex bytes...?\n";
|
|
}
|
|
} while (@_ > 0);
|
|
printf(" )\n");
|
|
}
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'z' and 'Z' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_bp_wp_command
|
|
{
|
|
my $cmd = shift;
|
|
my $type = shift;
|
|
shift; # Skip ','
|
|
my $address = get_addr(\@_);
|
|
shift; # Skip ','
|
|
my $length = join('',@_);
|
|
if ($cmd eq 'z')
|
|
{
|
|
printf("remove $point_types[$type]($addr_format, %d)\n", $address, $length);
|
|
}
|
|
else
|
|
{
|
|
printf("insert $point_types[$type]($addr_format, %d)\n", $address, $length);
|
|
}
|
|
}
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'X' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_write_mem_binary_cmd
|
|
{
|
|
my $cmd = shift;
|
|
my $address = get_addr(\@_);
|
|
shift; # Skip ','
|
|
|
|
my ($length, $binary) = split(/:/, join('',@_));
|
|
printf("write_mem_binary ( $addr_format, %d, %s)\n", $address, $length, $binary);
|
|
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'M' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_write_mem_cmd
|
|
{
|
|
my $cmd = shift;
|
|
my $address = get_addr(\@_);
|
|
shift; # Skip ','
|
|
my ($length, $hex_bytes) = split(/:/, join('',@_));
|
|
# printf("write_mem ( $addr_format, %d, %s)\n", $address, $length, $hex_bytes);
|
|
printf("write_mem ( addr = $addr_format, len = %d (0x%x), bytes = ", $address, $length, $length);
|
|
splice(@_, 0, length($length)+1);
|
|
|
|
my $curr_address = $address;
|
|
my $nibble;
|
|
my $nibble_count = 0;
|
|
my $max_nibbles_per_line = 2 * $max_bytes_per_line;
|
|
foreach $nibble (@_)
|
|
{
|
|
(($nibble_count % 2) == 0) and print ' ';
|
|
print $nibble;
|
|
$nibble_count++;
|
|
}
|
|
|
|
# If the memory to write is 2 or 4 bytes, print it out in native format
|
|
# instead of just as bytes.
|
|
if (@_ == 4)
|
|
{
|
|
printf(" ( 0x%4.4x )", get16(\@_));
|
|
}
|
|
elsif (@_ == 8)
|
|
{
|
|
printf(" ( 0x%8.8x )", get32(\@_));
|
|
}
|
|
print " )\n";
|
|
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'v' command
|
|
#----------------------------------------------------------------------
|
|
our $extended_rsp_callback = 0;
|
|
sub dump_extended_cmd
|
|
{
|
|
$extended_rsp_callback = 0;
|
|
if (join('', @_[0..4]) eq "vCont")
|
|
{
|
|
dump_extended_continue_cmd(splice(@_,5));
|
|
}
|
|
elsif (join('', @_[0..7]) eq 'vAttach;')
|
|
{
|
|
dump_attach_command (splice(@_,8));
|
|
}
|
|
elsif (join('', @_[0..11]) eq 'vAttachWait;')
|
|
{
|
|
dump_attach_wait_command (splice(@_,12));
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'v' response
|
|
#----------------------------------------------------------------------
|
|
sub dump_extended_rsp
|
|
{
|
|
if ($extended_rsp_callback)
|
|
{
|
|
&$extended_rsp_callback(@_);
|
|
}
|
|
$extended_rsp_callback = 0;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'vAttachWait' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_attach_wait_command
|
|
{
|
|
print "attach_wait ( ";
|
|
while (@_)
|
|
{
|
|
printf("%c", get8(\@_))
|
|
}
|
|
printf " )\n";
|
|
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'vAttach' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_attach_command
|
|
{
|
|
printf("attach ( pid = %i )", get_hex(\@_));
|
|
$extended_rsp_callback = \&dump_stop_reply_packet;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'vCont' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_extended_continue_cmd
|
|
{
|
|
print "extended_continue ( ";
|
|
my $cmd = shift;
|
|
if ($cmd eq '?')
|
|
{
|
|
print "list supported modes )\n";
|
|
$extended_rsp_callback = \&dump_extended_continue_rsp;
|
|
}
|
|
elsif ($cmd eq ';')
|
|
{
|
|
$extended_rsp_callback = \&dump_stop_reply_packet;
|
|
my $i = 0;
|
|
while ($#_ >= 0)
|
|
{
|
|
if ($i > 0)
|
|
{
|
|
print ", ";
|
|
}
|
|
my $continue_cmd = shift;
|
|
my $tmp;
|
|
if ($continue_cmd eq 'c')
|
|
{
|
|
print "continue";
|
|
}
|
|
elsif ($continue_cmd eq 'C')
|
|
{
|
|
print "continue with signal ";
|
|
print shift;
|
|
print shift;
|
|
}
|
|
elsif ($continue_cmd eq 's')
|
|
{
|
|
print "step";
|
|
}
|
|
elsif ($continue_cmd eq 'S')
|
|
{
|
|
print "step with signal ";
|
|
print shift;
|
|
print shift;
|
|
}
|
|
|
|
if ($_[0] eq ':')
|
|
{
|
|
shift; # Skip ':'
|
|
print " for thread ";
|
|
while ($#_ >= 0)
|
|
{
|
|
$tmp = shift;
|
|
if (length($tmp) > 0 && $tmp ne ';') {
|
|
print $tmp;
|
|
} else {
|
|
last;
|
|
}
|
|
}
|
|
}
|
|
$i++;
|
|
}
|
|
|
|
printf " )\n";
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# 'vCont' response
|
|
#----------------------------------------------------------------------
|
|
sub dump_extended_continue_rsp
|
|
{
|
|
if (scalar(@_) == 0)
|
|
{
|
|
print "$unimplemented_str\n";
|
|
}
|
|
else
|
|
{
|
|
print "extended_continue supports " . join('',@_) . "\n";
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Dump the command ascii for any unknown commands
|
|
#----------------------------------------------------------------------
|
|
sub dump_other_cmd
|
|
{
|
|
print "other = " . join('',@_) . "\n";
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Check to see if the response was unsupported with appropriate checksum
|
|
#----------------------------------------------------------------------
|
|
sub rsp_is_unsupported
|
|
{
|
|
return join('',@_) eq "#00";
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Check to see if the response was "OK" with appropriate checksum
|
|
#----------------------------------------------------------------------
|
|
sub rsp_is_OK
|
|
{
|
|
return join('',@_) eq "OK#9a";
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Dump a response for an unknown command
|
|
#----------------------------------------------------------------------
|
|
sub dump_other_rsp
|
|
{
|
|
print "other = " . join('',@_) . "\n";
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Get a byte from the ascii string assuming that the 2 nibble ascii
|
|
# characters are in hex.
|
|
#
|
|
# The argument for this function needs to be a reference to an array
|
|
# that contains single character strings and the array will get
|
|
# updated by shifting characters off the front of it (no leading # "0x")
|
|
#----------------------------------------------------------------------
|
|
sub get8
|
|
{
|
|
my $arrayref = shift;
|
|
my $val = hex(shift(@$arrayref) . shift(@$arrayref));
|
|
return $val;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Get a 16 bit integer and swap if $swap global is set to a non-zero
|
|
# value.
|
|
#
|
|
# The argument for this function needs to be a reference to an array
|
|
# that contains single character strings and the array will get
|
|
# updated by shifting characters off the front of it (no leading # "0x")
|
|
#----------------------------------------------------------------------
|
|
sub get16
|
|
{
|
|
my $arrayref = shift;
|
|
my $val = 0;
|
|
if ($swap)
|
|
{
|
|
$val = get8($arrayref) |
|
|
get8($arrayref) << 8;
|
|
}
|
|
else
|
|
{
|
|
$val = get8($arrayref) << 8 |
|
|
get8($arrayref) ;
|
|
}
|
|
return $val;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Get a 32 bit integer and swap if $swap global is set to a non-zero
|
|
# value.
|
|
#
|
|
# The argument for this function needs to be a reference to an array
|
|
# that contains single character strings and the array will get
|
|
# updated by shifting characters off the front of it (no leading # "0x")
|
|
#----------------------------------------------------------------------
|
|
sub get32
|
|
{
|
|
my $arrayref = shift;
|
|
my $val = 0;
|
|
if ($swap)
|
|
{
|
|
$val = get8($arrayref) |
|
|
get8($arrayref) << 8 |
|
|
get8($arrayref) << 16 |
|
|
get8($arrayref) << 24 ;
|
|
}
|
|
else
|
|
{
|
|
$val = get8($arrayref) << 24 |
|
|
get8($arrayref) << 16 |
|
|
get8($arrayref) << 8 |
|
|
get8($arrayref) ;
|
|
}
|
|
return $val;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Get a 64 bit hex value as a string
|
|
#
|
|
# The argument for this function needs to be a reference to an array
|
|
# that contains single character strings and the array will get
|
|
# updated by shifting characters off the front of it (no leading # "0x")
|
|
#----------------------------------------------------------------------
|
|
sub get64
|
|
{
|
|
my $arrayref = shift;
|
|
my $val = '';
|
|
my @nibbles;
|
|
if ($swap)
|
|
{
|
|
push @nibbles, splice(@$arrayref, 14, 2);
|
|
push @nibbles, splice(@$arrayref, 12, 2);
|
|
push @nibbles, splice(@$arrayref, 10, 2);
|
|
push @nibbles, splice(@$arrayref, 8, 2);
|
|
push @nibbles, splice(@$arrayref, 6, 2);
|
|
push @nibbles, splice(@$arrayref, 4, 2);
|
|
push @nibbles, splice(@$arrayref, 2, 2);
|
|
push @nibbles, splice(@$arrayref, 0, 2);
|
|
}
|
|
else
|
|
{
|
|
(@nibbles) = splice(@$arrayref, 0, ((64/8) * 2));
|
|
}
|
|
$val = join('', @nibbles);
|
|
return $val;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Get a 80 bit hex value as a string
|
|
#
|
|
# The argument for this function needs to be a reference to an array
|
|
# that contains single character strings and the array will get
|
|
# updated by shifting characters off the front of it (no leading # "0x")
|
|
#----------------------------------------------------------------------
|
|
sub get80
|
|
{
|
|
my $arrayref = shift;
|
|
my $val = '';
|
|
my @nibbles;
|
|
if ($swap)
|
|
{
|
|
push @nibbles, splice(@$arrayref, 18, 2);
|
|
push @nibbles, splice(@$arrayref, 16, 2);
|
|
push @nibbles, splice(@$arrayref, 14, 2);
|
|
push @nibbles, splice(@$arrayref, 12, 2);
|
|
push @nibbles, splice(@$arrayref, 10, 2);
|
|
push @nibbles, splice(@$arrayref, 8, 2);
|
|
push @nibbles, splice(@$arrayref, 6, 2);
|
|
push @nibbles, splice(@$arrayref, 4, 2);
|
|
push @nibbles, splice(@$arrayref, 2, 2);
|
|
push @nibbles, splice(@$arrayref, 0, 2);
|
|
}
|
|
else
|
|
{
|
|
(@nibbles) = splice(@$arrayref, 0, ((80/8) * 2));
|
|
}
|
|
$val = join('', @nibbles);
|
|
return $val;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Get a 96 bit hex value as a string
|
|
#
|
|
# The argument for this function needs to be a reference to an array
|
|
# that contains single character strings and the array will get
|
|
# updated by shifting characters off the front of it (no leading # "0x")
|
|
#----------------------------------------------------------------------
|
|
sub get96
|
|
{
|
|
my $arrayref = shift;
|
|
my $val = '';
|
|
my @nibbles;
|
|
if ($swap)
|
|
{
|
|
push @nibbles, splice(@$arrayref, 22, 2);
|
|
push @nibbles, splice(@$arrayref, 20, 2);
|
|
push @nibbles, splice(@$arrayref, 18, 2);
|
|
push @nibbles, splice(@$arrayref, 16, 2);
|
|
push @nibbles, splice(@$arrayref, 14, 2);
|
|
push @nibbles, splice(@$arrayref, 12, 2);
|
|
push @nibbles, splice(@$arrayref, 10, 2);
|
|
push @nibbles, splice(@$arrayref, 8, 2);
|
|
push @nibbles, splice(@$arrayref, 6, 2);
|
|
push @nibbles, splice(@$arrayref, 4, 2);
|
|
push @nibbles, splice(@$arrayref, 2, 2);
|
|
push @nibbles, splice(@$arrayref, 0, 2);
|
|
}
|
|
else
|
|
{
|
|
(@nibbles) = splice(@$arrayref, 0, ((96/8) * 2));
|
|
}
|
|
$val = join('', @nibbles);
|
|
return $val;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Get a 128 bit hex value as a string
|
|
#
|
|
# The argument for this function needs to be a reference to an array
|
|
# that contains single character strings and the array will get
|
|
# updated by shifting characters off the front of it (no leading # "0x")
|
|
#----------------------------------------------------------------------
|
|
sub get128
|
|
{
|
|
my $arrayref = shift;
|
|
my $val = '';
|
|
my @nibbles;
|
|
if ($swap)
|
|
{
|
|
push @nibbles, splice(@$arrayref, 30, 2);
|
|
push @nibbles, splice(@$arrayref, 28, 2);
|
|
push @nibbles, splice(@$arrayref, 26, 2);
|
|
push @nibbles, splice(@$arrayref, 24, 2);
|
|
push @nibbles, splice(@$arrayref, 22, 2);
|
|
push @nibbles, splice(@$arrayref, 20, 2);
|
|
push @nibbles, splice(@$arrayref, 18, 2);
|
|
push @nibbles, splice(@$arrayref, 16, 2);
|
|
push @nibbles, splice(@$arrayref, 14, 2);
|
|
push @nibbles, splice(@$arrayref, 12, 2);
|
|
push @nibbles, splice(@$arrayref, 10, 2);
|
|
push @nibbles, splice(@$arrayref, 8, 2);
|
|
push @nibbles, splice(@$arrayref, 6, 2);
|
|
push @nibbles, splice(@$arrayref, 4, 2);
|
|
push @nibbles, splice(@$arrayref, 2, 2);
|
|
push @nibbles, splice(@$arrayref, 0, 2);
|
|
}
|
|
else
|
|
{
|
|
(@nibbles) = splice(@$arrayref, 0, ((128/8) * 2));
|
|
}
|
|
$val = join('', @nibbles);
|
|
return $val;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Get a 256 bit hex value as a string
|
|
#
|
|
# The argument for this function needs to be a reference to an array
|
|
# that contains single character strings and the array will get
|
|
# updated by shifting characters off the front of it (no leading # "0x")
|
|
#----------------------------------------------------------------------
|
|
sub get256
|
|
{
|
|
my $arrayref = shift;
|
|
my $val = '';
|
|
my @nibbles;
|
|
if ($swap)
|
|
{
|
|
push @nibbles, splice(@$arrayref, 62, 2);
|
|
push @nibbles, splice(@$arrayref, 60, 2);
|
|
push @nibbles, splice(@$arrayref, 58, 2);
|
|
push @nibbles, splice(@$arrayref, 56, 2);
|
|
push @nibbles, splice(@$arrayref, 54, 2);
|
|
push @nibbles, splice(@$arrayref, 52, 2);
|
|
push @nibbles, splice(@$arrayref, 50, 2);
|
|
push @nibbles, splice(@$arrayref, 48, 2);
|
|
push @nibbles, splice(@$arrayref, 46, 2);
|
|
push @nibbles, splice(@$arrayref, 44, 2);
|
|
push @nibbles, splice(@$arrayref, 42, 2);
|
|
push @nibbles, splice(@$arrayref, 40, 2);
|
|
push @nibbles, splice(@$arrayref, 38, 2);
|
|
push @nibbles, splice(@$arrayref, 36, 2);
|
|
push @nibbles, splice(@$arrayref, 34, 2);
|
|
push @nibbles, splice(@$arrayref, 32, 2);
|
|
push @nibbles, splice(@$arrayref, 30, 2);
|
|
push @nibbles, splice(@$arrayref, 28, 2);
|
|
push @nibbles, splice(@$arrayref, 26, 2);
|
|
push @nibbles, splice(@$arrayref, 24, 2);
|
|
push @nibbles, splice(@$arrayref, 22, 2);
|
|
push @nibbles, splice(@$arrayref, 20, 2);
|
|
push @nibbles, splice(@$arrayref, 18, 2);
|
|
push @nibbles, splice(@$arrayref, 16, 2);
|
|
push @nibbles, splice(@$arrayref, 14, 2);
|
|
push @nibbles, splice(@$arrayref, 12, 2);
|
|
push @nibbles, splice(@$arrayref, 10, 2);
|
|
push @nibbles, splice(@$arrayref, 8, 2);
|
|
push @nibbles, splice(@$arrayref, 6, 2);
|
|
push @nibbles, splice(@$arrayref, 4, 2);
|
|
push @nibbles, splice(@$arrayref, 2, 2);
|
|
push @nibbles, splice(@$arrayref, 0, 2);
|
|
}
|
|
else
|
|
{
|
|
(@nibbles) = splice(@$arrayref, 0, ((256/8) * 2));
|
|
}
|
|
$val = join('', @nibbles);
|
|
return $val;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Get an unsigned integer value by grabbing items off the front of
|
|
# the array stopping when a non-digit char string is encountered.
|
|
#
|
|
# The argument for this function needs to be a reference to an array
|
|
# that contains single character strings and the array will get
|
|
# updated by shifting characters off the front of it
|
|
#----------------------------------------------------------------------
|
|
sub get_uint
|
|
{
|
|
my $arrayref = shift;
|
|
@$arrayref == 0 and return 0;
|
|
my $val = 0;
|
|
while ($$arrayref[0] =~ /[0-9]/)
|
|
{
|
|
$val = $val * 10 + int(shift(@$arrayref));
|
|
}
|
|
return $val;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Check the first character in the array and if it matches the expected
|
|
# character, return that character, else return undef;
|
|
#
|
|
# The argument for this function needs to be a reference to an array
|
|
# that contains single character strings and the array will get
|
|
# updated by shifting characters off the front of it. If the expected
|
|
# character doesn't match, it won't touch the array. If the first
|
|
# character does match, it will shift it off and return it.
|
|
#----------------------------------------------------------------------
|
|
sub get_expected_char
|
|
{
|
|
my $arrayref = shift;
|
|
my $expected_char = shift;
|
|
if ($expected_char eq $$arrayref[0])
|
|
{
|
|
return shift(@$arrayref);
|
|
}
|
|
return undef;
|
|
}
|
|
#----------------------------------------------------------------------
|
|
# Get a hex value by grabbing items off the front of the array and
|
|
# stopping when a non-hex char string is encountered.
|
|
#
|
|
# The argument for this function needs to be a reference to an array
|
|
# that contains single character strings and the array will get
|
|
# updated by shifting characters off the front of it (no leading # "0x")
|
|
#----------------------------------------------------------------------
|
|
sub get_hex
|
|
{
|
|
my $arrayref = shift;
|
|
my $my_swap = @_ ? shift : 0;
|
|
my $shift = 0;
|
|
my $val = 0;
|
|
while ($$arrayref[0] =~ /[0-9a-fA-F]/)
|
|
{
|
|
if ($my_swap)
|
|
{
|
|
my $byte = hex(shift(@$arrayref)) << 4 | hex(shift(@$arrayref));
|
|
$val |= $byte << $shift;
|
|
$shift += 8;
|
|
}
|
|
else
|
|
{
|
|
$val <<= 4;
|
|
$val |= hex(shift(@$arrayref));
|
|
}
|
|
}
|
|
return $val;
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Get an address value by grabbing items off the front of the array.
|
|
#
|
|
# The argument for this function needs to be a reference to an array
|
|
# that contains single character strings and the array will get
|
|
# updated by shifting characters off the front of it (no leading # "0x")
|
|
#----------------------------------------------------------------------
|
|
sub get_addr
|
|
{
|
|
get_hex(shift);
|
|
}
|
|
|
|
sub get_hex_string
|
|
{
|
|
my $arrayref = shift;
|
|
my $str = '';
|
|
while ($$arrayref[0] =~ /[0-9a-fA-F]/ and $$arrayref[1] =~ /[0-9a-fA-F]/)
|
|
{
|
|
my $hi_nibble = hex(shift(@$arrayref));
|
|
my $lo_nibble = hex(shift(@$arrayref));
|
|
my $byte = ($hi_nibble << 4) | $lo_nibble;
|
|
$str .= chr($byte);
|
|
}
|
|
return $str;
|
|
}
|
|
|
|
sub dump_stop_reply_data
|
|
{
|
|
while ($#_ >= 0)
|
|
{
|
|
last unless ($_[0] ne '#');
|
|
|
|
|
|
my $key = '';
|
|
my $value = '';
|
|
my $comment = '';
|
|
if ($_[0] =~ /[0-9a-fA-F]/ && $_[1] =~ /[0-9a-fA-F]/)
|
|
{
|
|
my $reg_num = get8(\@_);
|
|
shift(@_); # Skip ':'
|
|
if (defined ($registers_aref) && $reg_num < @$registers_aref)
|
|
{
|
|
dump_register_value(1, \@_, $reg_num);
|
|
print "\n";
|
|
shift(@_); # Skip ';'
|
|
next;
|
|
}
|
|
$key = sprintf("reg %u", $reg_num);
|
|
}
|
|
my $char;
|
|
|
|
if (length($key) == 0)
|
|
{
|
|
while (1)
|
|
{
|
|
$char = shift(@_);
|
|
if (length($char) == 0 or $char eq ':' or $char eq '#') { last; }
|
|
$key .= $char;
|
|
}
|
|
}
|
|
|
|
while (1)
|
|
{
|
|
$char = shift(@_);
|
|
if (length($char) == 0 or $char eq ';' or $char eq '#') { last; }
|
|
$value .= $char;
|
|
}
|
|
if ($key eq 'metype')
|
|
{
|
|
our %metype_to_name = (
|
|
'1' => ' (EXC_BAD_ACCESS)',
|
|
'2' => ' (EXC_BAD_INSTRUCTION)',
|
|
'3' => ' (EXC_ARITHMETIC)',
|
|
'4' => ' (EXC_EMULATION)',
|
|
'5' => ' (EXC_SOFTWARE)',
|
|
'6' => ' (EXC_BREAKPOINT)',
|
|
'7' => ' (EXC_SYSCALL)',
|
|
'8' => ' (EXC_MACH_SYSCALL)',
|
|
'9' => ' (EXC_RPC_ALERT)',
|
|
'10' => ' (EXC_CRASH)'
|
|
);
|
|
if (exists $metype_to_name{$value})
|
|
{
|
|
$comment = $metype_to_name{$value};
|
|
}
|
|
}
|
|
printf("\t%*s = %s$comment\n", $max_register_name_len, $key, $value);
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# Dumps a Stop Reply Packet which happens in response to a step,
|
|
# continue, last signal, and probably a few other commands.
|
|
#----------------------------------------------------------------------
|
|
sub dump_stop_reply_packet
|
|
{
|
|
my $what = shift(@_);
|
|
if ($what eq 'S' or $what eq 'T')
|
|
{
|
|
my $signo = get8(\@_);
|
|
|
|
our %signo_to_name = (
|
|
'1' => ' SIGHUP',
|
|
'2' => ' SIGINT',
|
|
'3' => ' SIGQUIT',
|
|
'4' => ' SIGILL',
|
|
'5' => ' SIGTRAP',
|
|
'6' => ' SIGABRT',
|
|
'7' => ' SIGPOLL/SIGEMT',
|
|
'8' => ' SIGFPE',
|
|
'9' => ' SIGKILL',
|
|
'10' => ' SIGBUS',
|
|
'11' => ' SIGSEGV',
|
|
'12' => ' SIGSYS',
|
|
'13' => ' SIGPIPE',
|
|
'14' => ' SIGALRM',
|
|
'15' => ' SIGTERM',
|
|
'16' => ' SIGURG',
|
|
'17' => ' SIGSTOP',
|
|
'18' => ' SIGTSTP',
|
|
'19' => ' SIGCONT',
|
|
'20' => ' SIGCHLD',
|
|
'21' => ' SIGTTIN',
|
|
'22' => ' SIGTTOU',
|
|
'23' => ' SIGIO',
|
|
'24' => ' SIGXCPU',
|
|
'25' => ' SIGXFSZ',
|
|
'26' => ' SIGVTALRM',
|
|
'27' => ' SIGPROF',
|
|
'28' => ' SIGWINCH',
|
|
'29' => ' SIGINFO',
|
|
'30' => ' SIGUSR1',
|
|
'31' => ' SIGUSR2',
|
|
'145' => ' TARGET_EXC_BAD_ACCESS', # 0x91
|
|
'146' => ' TARGET_EXC_BAD_INSTRUCTION', # 0x92
|
|
'147' => ' TARGET_EXC_ARITHMETIC', # 0x93
|
|
'148' => ' TARGET_EXC_EMULATION', # 0x94
|
|
'149' => ' TARGET_EXC_SOFTWARE', # 0x95
|
|
'150' => ' TARGET_EXC_BREAKPOINT' # 0x96
|
|
);
|
|
my $signo_str = sprintf("%i", $signo);
|
|
my $signo_name = '';
|
|
if (exists $signo_to_name{$signo_str})
|
|
{
|
|
$signo_name = $signo_to_name{$signo_str};
|
|
}
|
|
printf ("signal (signo=%u$signo_name)\n", $signo);
|
|
dump_stop_reply_data (@_);
|
|
}
|
|
elsif ($what eq 'W')
|
|
{
|
|
print 'process_exited( ' . shift(@_) . shift(@_) . " )\n";
|
|
}
|
|
elsif ($what eq 'X')
|
|
{
|
|
print 'process_terminated( ' . shift(@_) . shift(@_) . " )\n";
|
|
}
|
|
elsif ($what eq 'O')
|
|
{
|
|
my $console_output = '';
|
|
my $num_hex8_bytes = @_/2;
|
|
for (1 .. $num_hex8_bytes)
|
|
{
|
|
$console_output .= sprintf("%c", get8(\@_))
|
|
}
|
|
|
|
print "program_console_output('$console_output')\n";
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# '?' command
|
|
#----------------------------------------------------------------------
|
|
sub dump_last_signal_cmd
|
|
{
|
|
my $cmd = shift;
|
|
print 'last_signal (' . join('',@_) . ")\n";
|
|
}
|
|
|
|
sub dump_raw_command
|
|
{
|
|
my $cmd_aref = shift;
|
|
my $callback_ref;
|
|
$curr_cmd = $$cmd_aref[0];
|
|
|
|
if ($curr_cmd eq 'q' or $curr_cmd eq 'Q' or $curr_cmd eq '_')
|
|
{
|
|
$curr_full_cmd = '';
|
|
foreach my $ch (@$cmd_aref)
|
|
{
|
|
$ch !~ /[A-Za-z_]/ and last;
|
|
$curr_full_cmd .= $ch;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
$curr_full_cmd = $curr_cmd;
|
|
}
|
|
|
|
$curr_cmd eq '_' and $curr_cmd .= $$cmd_aref[1];
|
|
$callback_ref = $cmd_callbacks{$curr_cmd};
|
|
if ($callback_ref)
|
|
{
|
|
&$callback_ref(@$cmd_aref);
|
|
}
|
|
else
|
|
{
|
|
# Strip the command byte for responses since we injected that above
|
|
dump_other_cmd(@$cmd_aref);
|
|
}
|
|
}
|
|
|
|
sub dump_standard_response
|
|
{
|
|
my $cmd_aref = shift;
|
|
|
|
my $cmd_len = scalar(@$cmd_aref);
|
|
if ($cmd_len == 0)
|
|
{
|
|
print "$unimplemented_str\n";
|
|
return 1;
|
|
}
|
|
|
|
my $response = join('', @$cmd_aref);
|
|
if ($response eq 'OK')
|
|
{
|
|
print "$success_str\n";
|
|
return 1;
|
|
}
|
|
|
|
if ($cmd_len == 3 and index($response, 'E') == 0)
|
|
{
|
|
print "ERROR: " . substr($response, 1) . "\n";
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
sub dump_raw_response
|
|
{
|
|
my $cmd_aref = shift;
|
|
my $callback_ref;
|
|
|
|
if ($packet_start_time != 0.0)
|
|
{
|
|
if (length($curr_full_cmd) > 0)
|
|
{
|
|
$packet_times{$curr_full_cmd} += $curr_time - $packet_start_time;
|
|
}
|
|
else
|
|
{
|
|
$packet_times{$curr_cmd} += $curr_time - $packet_start_time;
|
|
}
|
|
$packet_start_time = 0.0;
|
|
}
|
|
|
|
$callback_ref = $rsp_callbacks{$curr_cmd};
|
|
|
|
if ($callback_ref)
|
|
{
|
|
&$callback_ref(@$cmd_aref);
|
|
}
|
|
else
|
|
{
|
|
dump_standard_response($cmd_aref) or dump_other_rsp(@$cmd_aref);
|
|
}
|
|
|
|
}
|
|
#----------------------------------------------------------------------
|
|
# Dumps any command and handles simple error checking on the responses
|
|
# for commands that are unsupported or OK.
|
|
#----------------------------------------------------------------------
|
|
sub dump_command
|
|
{
|
|
my $cmd_str = shift;
|
|
|
|
# Dump the original command string if verbose is on
|
|
if ($opt_v)
|
|
{
|
|
print "dump_command($cmd_str)\n ";
|
|
}
|
|
|
|
my @cmd_chars = extract_command($cmd_str);
|
|
my $is_cmd = 1;
|
|
|
|
my $cmd = $cmd_chars[0];
|
|
if ($cmd eq '$')
|
|
{
|
|
$is_cmd = 0; # Note that this is a reply
|
|
$cmd = $curr_cmd; # set the command byte appropriately
|
|
shift @cmd_chars; # remove the '$' from the cmd bytes
|
|
}
|
|
|
|
# Check for common responses across all commands and handle them
|
|
# if we can
|
|
if ( $is_cmd == 0 )
|
|
{
|
|
if (rsp_is_unsupported(@cmd_chars))
|
|
{
|
|
print "$unimplemented_str\n";
|
|
return;
|
|
}
|
|
elsif (rsp_is_OK(@cmd_chars))
|
|
{
|
|
print "$success_str\n";
|
|
return;
|
|
}
|
|
# Strip the checksum information for responses
|
|
strip_checksum(\@cmd_chars);
|
|
}
|
|
|
|
my $callback_ref;
|
|
if ($is_cmd) {
|
|
$callback_ref = $cmd_callbacks{$cmd};
|
|
} else {
|
|
$callback_ref = $rsp_callbacks{$cmd};
|
|
}
|
|
|
|
if ($callback_ref)
|
|
{
|
|
&$callback_ref(@cmd_chars);
|
|
}
|
|
else
|
|
{
|
|
# Strip the command byte for responses since we injected that above
|
|
if ($is_cmd) {
|
|
dump_other_cmd(@cmd_chars);
|
|
} else {
|
|
dump_other_rsp(@cmd_chars);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
# Process a gdbserver log line by looking for getpkt and putkpt and
|
|
# tossing any other lines.
|
|
|
|
#----------------------------------------------------------------------
|
|
sub process_log_line
|
|
{
|
|
my $line = shift;
|
|
#($opt_v and $opt_g) and print "# $line";
|
|
|
|
my $extract_cmd = 0;
|
|
my $delta_time = 0.0;
|
|
if ($line =~ /^(\s*)([1-9][0-9]+\.[0-9]+)([^0-9].*)$/)
|
|
{
|
|
my $leading_space = $1;
|
|
$curr_time = $2;
|
|
$line = $3;
|
|
if ($base_time == 0.0)
|
|
{
|
|
$base_time = $curr_time;
|
|
}
|
|
else
|
|
{
|
|
$delta_time = $curr_time - $last_time;
|
|
}
|
|
printf ("(%.6f, %+.6f): ", $curr_time - $base_time, $delta_time);
|
|
$last_time = $curr_time;
|
|
}
|
|
else
|
|
{
|
|
$curr_time = 0.0
|
|
}
|
|
|
|
if ($line =~ /getpkt /)
|
|
{
|
|
$extract_cmd = 1;
|
|
print "\n--> ";
|
|
$packet_start_time = $curr_time;
|
|
}
|
|
elsif ($line =~ /putpkt /)
|
|
{
|
|
$extract_cmd = 1;
|
|
print "<-- ";
|
|
}
|
|
elsif ($line =~ /.*Sent: \[[0-9]+\.[0-9]+[:0-9]*\] (.*)/)
|
|
{
|
|
$opt_g and print "maintenance dump-packets command: $1\n";
|
|
my @raw_cmd_bytes = split(/ */, $1);
|
|
$packet_start_time = $curr_time;
|
|
print "\n--> ";
|
|
dump_raw_command(\@raw_cmd_bytes);
|
|
process_log_line($2);
|
|
}
|
|
elsif ($line =~ /.*Recvd: \[[0-9]+\.[0-9]+[:0-9]*\] (.*)/)
|
|
{
|
|
$opt_g and print "maintenance dump-packets reply: $1\n";
|
|
my @raw_rsp_bytes = split(/ */, $1);
|
|
print "<-- ";
|
|
dump_raw_response(\@raw_rsp_bytes);
|
|
print "\n";
|
|
}
|
|
elsif ($line =~ /getpkt: (.*)/)
|
|
{
|
|
if ($1 =~ /\$([^#]+)#[0-9a-fA-F]{2}/)
|
|
{
|
|
$opt_g and print "command: $1\n";
|
|
my @raw_cmd_bytes = split(/ */, $1);
|
|
print "--> ";
|
|
$packet_start_time = $curr_time;
|
|
dump_raw_command(\@raw_cmd_bytes);
|
|
}
|
|
elsif ($1 =~ /\+/)
|
|
{
|
|
#print "--> ACK\n";
|
|
}
|
|
elsif ($1 =~ /-/)
|
|
{
|
|
#print "--> NACK\n";
|
|
}
|
|
}
|
|
elsif ($line =~ /putpkt: (.*)/)
|
|
{
|
|
if ($1 =~ /\$([^#]+)#[0-9a-fA-F]{2}/)
|
|
{
|
|
$opt_g and print "response: $1\n";
|
|
my @raw_rsp_bytes = split(/ */, $1);
|
|
print "<-- ";
|
|
dump_raw_response(\@raw_rsp_bytes);
|
|
print "\n";
|
|
}
|
|
elsif ($1 =~ /\+/)
|
|
{
|
|
#print "<-- ACK\n";
|
|
}
|
|
elsif ($1 =~ /-/)
|
|
{
|
|
#print "<-- NACK\n";
|
|
}
|
|
}
|
|
elsif ($line =~ /send packet: (.*)/)
|
|
{
|
|
if ($1 =~ /\$([^#]+)#[0-9a-fA-F]{2}/)
|
|
{
|
|
$opt_g and print "command: $1\n";
|
|
my @raw_cmd_bytes = split(/ */, $1);
|
|
print "--> ";
|
|
$packet_start_time = $curr_time;
|
|
dump_raw_command(\@raw_cmd_bytes);
|
|
}
|
|
elsif ($1 =~ /\+/)
|
|
{
|
|
#print "--> ACK\n";
|
|
}
|
|
elsif ($1 =~ /-/)
|
|
{
|
|
#print "--> NACK\n";
|
|
}
|
|
}
|
|
elsif ($line =~ /read packet: (.*)/)
|
|
{
|
|
if ($1 =~ /\$([^#]*)#[0-9a-fA-F]{2}/)
|
|
{
|
|
$opt_g and print "response: $1\n";
|
|
my @raw_rsp_bytes = split(/ */, $1);
|
|
print "<-- ";
|
|
dump_raw_response(\@raw_rsp_bytes);
|
|
print "\n";
|
|
}
|
|
elsif ($1 =~ /\+/)
|
|
{
|
|
#print "<-- ACK\n";
|
|
}
|
|
elsif ($1 =~ /-/)
|
|
{
|
|
#print "<-- NACK\n";
|
|
}
|
|
}
|
|
elsif ($line =~ /Sending packet: \$([^#]+)#[0-9a-fA-F]{2}\.\.\.(.*)/)
|
|
{
|
|
$opt_g and print "command: $1\n";
|
|
my @raw_cmd_bytes = split(/ */, $1);
|
|
print "\n--> ";
|
|
$packet_start_time = $curr_time;
|
|
dump_raw_command(\@raw_cmd_bytes);
|
|
process_log_line($2);
|
|
}
|
|
elsif ($line =~ /Packet received: (.*)/)
|
|
{
|
|
$opt_g and print "response: $1\n";
|
|
my @raw_rsp_bytes = split(/ */, $1);
|
|
print "<-- ";
|
|
dump_raw_response(\@raw_rsp_bytes);
|
|
print "\n";
|
|
}
|
|
|
|
if ($extract_cmd)
|
|
{
|
|
my $beg = index($line, '("') + 2;
|
|
my $end = rindex($line, '");');
|
|
$packet_start_time = $curr_time;
|
|
dump_command(substr($line, $beg, $end - $beg));
|
|
}
|
|
}
|
|
|
|
|
|
our $line_num = 0;
|
|
while(<>)
|
|
{
|
|
$line_num++;
|
|
$opt_q or printf("# %5d: $_", $line_num);
|
|
process_log_line($_);
|
|
}
|
|
|
|
if (%packet_times)
|
|
{
|
|
print "----------------------------------------------------------------------\n";
|
|
print "Packet timing summary:\n";
|
|
print "----------------------------------------------------------------------\n";
|
|
print "Packet Time %\n";
|
|
print "---------------------- -------- ------\n";
|
|
my @packet_names = keys %packet_times;
|
|
my $total_packet_times = 0.0;
|
|
foreach my $key (@packet_names)
|
|
{
|
|
$total_packet_times += $packet_times{$key};
|
|
}
|
|
|
|
foreach my $value (sort {$packet_times{$b} cmp $packet_times{$a}} @packet_names)
|
|
{
|
|
my $percent = ($packet_times{$value} / $total_packet_times) * 100.0;
|
|
if ($percent < 10.0)
|
|
{
|
|
printf("%22s %1.6f %2.2f\n", $value, $packet_times{$value}, $percent);
|
|
|
|
}
|
|
else
|
|
{
|
|
printf("%22s %1.6f %2.2f\n", $value, $packet_times{$value}, $percent);
|
|
}
|
|
}
|
|
print "---------------------- -------- ------\n";
|
|
printf (" Total %1.6f 100.00\n", $total_packet_times);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|