# # \brief Test for using the GDB monitor # \author Christian Prochaska # \author Norman Feske # \date 2011-05-24 # # # Only the NOVA platform supports most of the tested features at this time. # if {![have_include "power_on/qemu"] || !([have_spec nova])} { puts "Run script is only supported for NOVA in Qemu"; exit 0 } # # Build # set build_components { core init timer drivers/uart app/gdb_monitor test/gdb_monitor } lappend build_components "lib/gdbserver_platform-$::env(KERNEL)" build $build_components create_boot_directory # # Generate config # set config { } install_config $config # # Boot modules # # evaluated by the run tool proc binary_name_gdbserver_platform_lib_so { } { return "gdbserver_platform-$::env(KERNEL).lib.so" } # generic modules set boot_modules { core init timer ld.lib.so libc.lib.so vfs.lib.so libm.lib.so libc_pipe.lib.so libc_terminal.lib.so uart_drv posix.lib.so gdb_monitor gdbserver_platform.lib.so test-gdb_monitor } build_boot_image $boot_modules # # Execute test case # # set local_port 5555 # qemu config append qemu_args " -nographic " # connect comport 0 to stdio append qemu_args " -serial mon:stdio " # connect comport 1 with TCP port $local_port append qemu_args " -serial chardev:uart " append qemu_args " -chardev socket,id=uart,port=$local_port,host=localhost,server,nowait,ipv4 " run_genode_until {.*\[init -> gdb_monitor\].*} 30 set genode_id [output_spawn_id] puts "GDB monitor is up, starting GDB" source ${genode_dir}/repos/ports/run/gdb_monitor.inc # GDB loads symbols from 'debug/ld.lib.so' if { [have_spec nova] } { exec ln -sf ld-nova.lib.so debug/ld.lib.so } set gdb_target_binary "test-gdb_monitor" # sequence of GDB commands to execute at startup set gdb_cmds "" append gdb_cmds {-ex "target remote localhost:$local_port" } append gdb_cmds [gdb_initial_breakpoint_cmds $gdb_target_binary] # run GDB and redirect stderr to stdio to get the relevant output into the expect buffer eval spawn [gdb] debug/ld.lib.so -n $gdb_cmds 2&>1 set gdb_id [list $spawn_id $genode_id] puts "" puts "----- test: breakpoint in 'main()' -----" puts "" run_genode_until {\(gdb\)} 60 $gdb_id send "b main\n" run_genode_until {\(gdb\)} 20 $gdb_id send "c\n" run_genode_until {\(gdb\)} 20 $gdb_id if {![regexp {Breakpoint 2, main ()} $output]} { puts stderr "*** Error: Breakpoint in main() did not trigger" exit -1 } send "delete 2\n" run_genode_until {\(gdb\)} 20 $gdb_id puts "\n" puts "----- test: breakpoint in shared library -----" puts "" send "b puts\n" run_genode_until {\(gdb\)} 20 $gdb_id send "c\n" run_genode_until {\(gdb\)} 20 $gdb_id if {![regexp {Breakpoint 3, puts ()} $output]} { puts "*** Error: Breakpoint in shared library did not trigger" exit -1 } puts "\n" puts "----- test: stack trace when not in syscall -----" puts "" send "bt\n" run_genode_until {\(gdb\)} 20 $gdb_id if {![regexp {#0 puts} $output] || ![regexp {in func2()} $output] || ![regexp {in func1()} $output] || ![regexp {in main ()} $output]} { puts stderr "*** Error: Stack trace when not in syscall is not as expected" exit -1 } puts "\n" puts "----- test: modification of a variable value -----" puts "" send "print test_var\n" run_genode_until {\(gdb\)} 20 $gdb_id if {![regexp {\$1 = 1} $output]} { puts stderr "*** Error: first 'print test_var' command didn't result in the expected output" exit -1 } send "set var test_var=2\n" run_genode_until {\(gdb\)} 20 $gdb_id send "print test_var\n" run_genode_until {\(gdb\)} 20 $gdb_id if {![regexp {\$2 = 2} $output]} { puts stderr "*** Error: second 'print test_var' command didn't result in the expected output" exit -1 } puts "\n" puts "----- test: 'call' command -----" puts "" send "call test_var_func()\n" run_genode_until {\(gdb\)} 60 $gdb_id if {![regexp {\$3 = 3} $output]} { puts stderr "*** Error: 'call' command didn't result in the expected output" exit -1 } puts "\n" puts "----- test: thread info -----" puts "" send "b test_thread_start\n" run_genode_until {\(gdb\)} 20 $gdb_id send "c\n" run_genode_until {\(gdb\)} 20 $gdb_id if {![regexp {Breakpoint 4, test_thread_start} $output]} { puts stderr "*** Error: Breakpoint in test thread did not trigger" exit -1 } send "info threads\n" run_genode_until {\(gdb\)} 20 $gdb_id if {![regexp { 4 Thread 3} $output] || ![regexp {\* 3 Thread 4 test_thread_start} $output] || ![regexp { 2 Thread 2} $output] || ![regexp { 1 Thread 1} $output]} { puts stderr "*** Error: Thread info is not as expected" exit -1 } puts "\n" puts "----- test: step into function -----" puts "" send "step\n" run_genode_until {\(gdb\)} 30 $gdb_id send "thread 2\n" run_genode_until {\(gdb\)} 20 $gdb_id if {![regexp {test_thread_step} $output]} { puts stderr "*** Error: Step into function didn't result in the expected output" exit -1 } puts "\n" puts "----- test: catching a segmentation fault -----" puts "" send "c\n" run_genode_until {\(gdb\)} 20 $gdb_id if {![regexp {Program received signal SIGSEGV, Segmentation fault.} $output]} { puts stderr "*** Error: Segmentation fault exception was not caught" exit -1 } # does not work well on ARM yet if {![have_spec arm]} { puts "\n" puts "----- test: stack trace when in syscall -----" puts "" send "thread 2\n" run_genode_until {\(gdb\)} 20 $gdb_id send "bt\n" run_genode_until {\(gdb\)} 20 $gdb_id if {![regexp {Genode::Cancelable_lock::lock\(\)} $output] || ![regexp {Genode::Signal_receiver::block_for_signal\(\)} $output] || ![regexp {Genode::Entrypoint::_wait_and_dispatch_one_io_signal\(bool\)} $output] || ![regexp {Libc::Kernel::run\(Libc::Application_code&\)} $output] } { puts stderr "*** Error: Stack trace when in syscall is not as expected" exit -1 } } puts "" # vi: set ft=tcl :