This is the mail archive of the gdb-patches@sourceware.org mailing list for the GDB project.


Index Nav: [Date Index] [Subject Index] [Author Index] [Thread Index]
Message Nav: [Date Prev] [Date Next] [Thread Prev] [Thread Next]
Other format: [Raw text]

[patch] [python] python/12686


Kevin Pouget noticed that when GDB called the Python 'stop' callback,
the terminal stilled belonged to the inferior.  GDB intercepts the
signal generated from terminal input and it forces GDB into the
background.  While this patch is simple, I would like feedback.  I
believe the terminal is restored to the inferior when the inferior is
resumed (local tests seem to show this), but I am not totally sure.  Is
acquiring the terminal the right thing to do here? I was trying to think
of a scenario where the Python method would need the inferior terminal,
but could not.

Comments?

Cheers,

Phil

--

2011-04-20  Phil Muldoon  <pmuldoon@redhat.com>

        PR python/12686

	* python/py-breakpoint.c (gdbpy_should_stop): Acquire terminal
	before executing Python code.

2011-04-20  Phil Muldoon  <pmuldoon@redhat.com>

        PR python/12686

	* gdb.python/py-breakpoint.exp: Add terminal input 'stop' method
	test.

--

diff --git a/gdb/python/py-breakpoint.c b/gdb/python/py-breakpoint.c
index 0c21bfc..3ee23be 100644
--- a/gdb/python/py-breakpoint.c
+++ b/gdb/python/py-breakpoint.c
@@ -729,7 +729,13 @@ gdbpy_should_stop (struct breakpoint_object *bp_obj)
 
   if (PyObject_HasAttrString (py_bp, stop_func))
     {
-      PyObject *result = PyObject_CallMethod (py_bp, stop_func, NULL);
+      PyObject *result;
+      
+      /* GDB acquires the terminal from the inferior before the
+	 call.  The Python 'stop' method might need to do
+	 input/output.  */
+      terminal_ours ();
+      result = PyObject_CallMethod (py_bp, stop_func, NULL);
 
       if (result)
 	{
diff --git a/gdb/testsuite/gdb.python/py-breakpoint.exp b/gdb/testsuite/gdb.python/py-breakpoint.exp
index f0a83f1..d46a37c 100644
--- a/gdb/testsuite/gdb.python/py-breakpoint.exp
+++ b/gdb/testsuite/gdb.python/py-breakpoint.exp
@@ -237,6 +237,15 @@ gdb_py_test_multiple "Sub-class a third breakpoint" \
   "   count = 0" "" \
   "end" ""
 
+gdb_py_test_multiple "Breakpoint that asks for input" \
+  "python" "" \
+  "class ask_bp_input (gdb.Breakpoint):" "" \
+  "    def stop (self):" "" \
+  "       inp = raw_input(\"Input\")" "" \
+  "       print inp" "" \
+  "       return True" "" \
+  "end" ""
+
 set bp_location2 [gdb_get_line_number "Break at multiply."]
 set end_location [gdb_get_line_number "Break at end."]
 gdb_py_test_silent_cmd  "python eval_bp1 = bp_eval(\"$bp_location2\")" "Set breakpoint" 0
@@ -293,3 +302,39 @@ gdb_py_test_silent_cmd  "python wp1 = wp_eval (\"result\", type=gdb.BP_WATCHPOIN
 gdb_test "continue" ".*\[Ww\]atchpoint.*result.*Old value =.*New value = 788.*" "Test watchpoint write"
 gdb_test "python print never_eval_bp1.count" "0" \
     "Check that this unrelated breakpoints eval function was never called."
+
+# Test breakpoints that require user input.
+# Start with a fresh gdb.
+delete_breakpoints
+clean_restart ${testfile}
+
+gdb_py_test_multiple "Breakpoint that asks for input" \
+  "python" "" \
+  "class ask_bp_input (gdb.Breakpoint):" "" \
+  "    def stop (self):" "" \
+  "       inp = raw_input(\"Input? \")" "" \
+  "       print \"Output =\", inp" "" \
+  "       return True" "" \
+  "end" ""
+
+set input_bp [gdb_get_line_number "Break at multiply."]
+gdb_py_test_silent_cmd  "python np_input = ask_bp_input(\"$input_bp\")" \
+    "Set breakpoint" 0
+send_gdb "run\n"
+gdb_expect {
+    -re "Input\?.*" {
+        pass "Input prompt has appeard and awaiting for input."
+    }
+    timeout {
+        fail "Input prompt has not appearead."
+    }
+}
+send_gdb "123\n"
+gdb_expect {
+    -re "Output = 123.*Breakpoint.*main.*py-breakpoint.*" {
+	pass "Output recieved from input is correct and printed."
+    }
+    timeout {
+        fail "Output not recieved from input prompt."
+    }
+}


Index Nav: [Date Index] [Subject Index] [Author Index] [Thread Index]
Message Nav: [Date Prev] [Date Next] [Thread Prev] [Thread Next]