| # Copyright (C) 2017-2021 Free Software Foundation, Inc. | 
 |  | 
 | # This program is free software; you can redistribute it and/or modify | 
 | # it under the terms of the GNU General Public License as published by | 
 | # the Free Software Foundation; either version 3 of the License, or | 
 | # (at your option) any later version. | 
 | # | 
 | # This program is distributed in the hope that it will be useful, | 
 | # but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
 | # GNU General Public License for more details. | 
 | # | 
 | # You should have received a copy of the GNU General Public License | 
 | # along with this program.  If not, see <http://www.gnu.org/licenses/>. | 
 |  | 
 | # Please email any bugs, comments, and/or additions to this file to: | 
 | # bug-gdb@gnu.org | 
 |  | 
 | # This file verifies that methods Inferior.thread_from_handle | 
 | # and InferiorThread.handle work as expected. | 
 |  | 
 | load_lib gdb-python.exp | 
 |  | 
 | standard_testfile | 
 |  | 
 | if {[gdb_compile_pthreads "${srcdir}/${subdir}/${srcfile}" "${binfile}" executable debug] != "" } { | 
 |     return -1 | 
 | } | 
 |  | 
 | clean_restart ${binfile} | 
 |  | 
 | # Skip all tests if Python scripting is not enabled. | 
 | if { [skip_python_tests] } { continue } | 
 |  | 
 | runto_main | 
 |  | 
 | gdb_test "break after_mc_barrier" \ | 
 |     "Breakpoint 2 at .*: file .*${srcfile}, line .*" \ | 
 |          "breakpoint on after_mc_barrier" | 
 |  | 
 | gdb_test "break do_something" \ | 
 |     "Breakpoint 3 at .*: file .*${srcfile}, line .*" \ | 
 |          "breakpoint on do_something" | 
 |  | 
 | gdb_test "continue" \ | 
 | 	"Breakpoint 2, after_mc_barrier .*" \ | 
 | 	"run to after_mc_barrier" | 
 |  | 
 | gdb_test_no_output "del 2" "delete after_mc_barrier breakpoint" | 
 |  | 
 | gdb_test "continue" \ | 
 | 	"Breakpoint 3, do_something .*" \ | 
 | 	"run to do_something" | 
 |  | 
 | # The test case has been constructed so that the current thread, | 
 | # indicated by '*' in the "info threads" output, should be stopped in | 
 | # do_something() with a value of n which is the same as the number | 
 | # reported in the "Id" column.  If it's not, then something went wrong | 
 | # with the start up sequence which should cause the main thread to be | 
 | # thread 1, the first child thread to be thread 2, and the second | 
 | # child thread to be thread 3. | 
 | # | 
 | # Note that \1 in the RE below is a backreference to the thread id | 
 | # reported in the "Id" column. | 
 |  | 
 | gdb_test "info threads"  \ | 
 | 	{.*[\r\n]+\* +([0-9]+) +Thread[^\r\n]* do_something \(n=\1\) at.*} | 
 |  | 
 | # Check for expected results when passing a valid thread handle to | 
 | # thread_from_handle(). | 
 |  | 
 | gdb_test "python print(gdb.selected_inferior().thread_from_handle(gdb.parse_and_eval('thrs\[0\]')).num)" \ | 
 | 	"1" "print thread id for thrs\[0\]" | 
 |  | 
 | gdb_test "python print(gdb.selected_inferior().thread_from_handle(gdb.parse_and_eval('thrs\[1\]')).num)" \ | 
 | 	"2" "print thread id for thrs\[1\]" | 
 |  | 
 | gdb_test "python print(gdb.selected_inferior().thread_from_handle(gdb.parse_and_eval('thrs\[2\]')).num)" \ | 
 | 	"3" "print thread id for thrs\[2\]" | 
 |  | 
 | # Objects which are of the correct size, but which are bogus thread | 
 | # handles should return None.  For the first test (using thrs[3]), we | 
 | # use 0.  For the second (thrs[4]), we use an unlikely bit pattern. | 
 |  | 
 | gdb_test "python print(gdb.selected_inferior().thread_from_handle(gdb.parse_and_eval('thrs\[3\]')))" \ | 
 | 	"None" "print thread for bogus handle thrs\[3\]" | 
 |  | 
 | gdb_test "python print(gdb.selected_inferior().thread_from_handle(gdb.parse_and_eval('thrs\[4\]')))" \ | 
 | 	"None" "print thread for bogus handle thrs\[4\]" | 
 |  | 
 | # We should see an exception when passing an object of the wrong type. | 
 |  | 
 | gdb_test "python print(gdb.selected_inferior().thread_from_handle(gdb.lookup_symbol('main')))" \ | 
 |          ".*TypeError: Argument 'handle' must be a thread handle object.*" \ | 
 | 	 "TypeError when passing a symbol object to thread_from_handle" | 
 |  | 
 | # We should see an exception when passing too large of an object. | 
 |  | 
 | gdb_test "python print(gdb.selected_inferior().thread_from_handle(gdb.parse_and_eval('thrs')))" \ | 
 |          ".*Thread handle size mismatch.*" \ | 
 | 	 "Pass overly large object to thread_from_handle" | 
 |  | 
 | # We should see an exception when passing too small of an object. | 
 |  | 
 | gdb_test "python print(gdb.selected_inferior().thread_from_handle(gdb.parse_and_eval('\"S\"')))" \ | 
 |          ".*Thread handle size mismatch.*" \ | 
 | 	 "Pass too small of an object to thread_from_handle" | 
 |  | 
 | # Test the thread_handle method | 
 |  | 
 | gdb_py_test_silent_cmd "python tp=gdb.lookup_type('pthread_t')" \ | 
 | 		       "Get pthread_t type" 0 | 
 | gdb_py_test_silent_cmd "python inf=gdb.selected_inferior()" "Get inferior" 0 | 
 |  | 
 | foreach thrN {0 1 2} { | 
 |     with_test_prefix "thread $thrN" { | 
 |  | 
 | 	gdb_py_test_silent_cmd \ | 
 | 	    "python hand = gdb.parse_and_eval('thrs\[$thrN\]')" \ | 
 | 	    "fetch thread handle from inferior" \ | 
 | 	    1 | 
 |  | 
 | 	gdb_py_test_silent_cmd \ | 
 | 	    "python hand_bytes = inf.thread_from_handle(hand).handle()" \ | 
 | 	    "fetch thread handle from thread" \ | 
 | 	    1 | 
 |  | 
 |  | 
 | 	# It'd be nice to be able to use this comparison expression: | 
 | 	# | 
 | 	#    hand == hand_bytes | 
 | 	# | 
 | 	# But this won't work because hand is a gdb.Value and hand_bytes | 
 | 	# is a Python bytes object.  Therefore, we convert the bytes | 
 | 	# object into a gdb.value by calling the two argument form of | 
 | 	# its constructor. | 
 |  | 
 |         gdb_test "python print(gdb.Value(hand_bytes, tp) == hand)" \ | 
 | 	         "True" \ | 
 | 	         "verify that handles are the same" | 
 |     } | 
 | } |