forked from OSchip/llvm-project
Add test cases where we start three worker threads, with a write watchpoint set. As soon as a watchpoint is hit in either of the worker thread,
we delete the watchpoint. The test succeeds when no more watchpoint hit event fires after the deletion of the watchpoint. related to rdar://problem/11320188 llvm-svn: 157084
This commit is contained in:
parent
4c82d425fa
commit
90830703b2
|
@ -4,6 +4,7 @@ Test that lldb watchpoint works for multiple threads.
|
|||
|
||||
import os, time
|
||||
import unittest2
|
||||
import re
|
||||
import lldb
|
||||
from lldbtest import *
|
||||
|
||||
|
@ -26,6 +27,21 @@ class WatchpointForMultipleThreadsTestCase(TestBase):
|
|||
self.setTearDownCleanup(dictionary=self.d)
|
||||
self.hello_multiple_threads()
|
||||
|
||||
@unittest2.skipUnless(sys.platform.startswith("darwin"), "requires Darwin")
|
||||
@dsym_test
|
||||
def test_watchpoint_multiple_threads_wp_set_and_then_delete_with_dsym(self):
|
||||
"""Test that lldb watchpoint works for multiple threads, and after the watchpoint is deleted, the watchpoint event should no longer fires."""
|
||||
self.buildDsym(dictionary=self.d)
|
||||
self.setTearDownCleanup(dictionary=self.d)
|
||||
self.hello_multiple_threads_wp_set_and_then_delete()
|
||||
|
||||
@dwarf_test
|
||||
def test_watchpoint_multiple_threads_wp_set_and_then_delete_with_dwarf(self):
|
||||
"""Test that lldb watchpoint works for multiple threads, and after the watchpoint is deleted, the watchpoint event should no longer fires."""
|
||||
self.buildDwarf(dictionary=self.d)
|
||||
self.setTearDownCleanup(dictionary=self.d)
|
||||
self.hello_multiple_threads_wp_set_and_then_delete()
|
||||
|
||||
def setUp(self):
|
||||
# Call super's setUp().
|
||||
TestBase.setUp(self)
|
||||
|
@ -81,6 +97,7 @@ class WatchpointForMultipleThreadsTestCase(TestBase):
|
|||
self.runCmd("thread list")
|
||||
if "stop reason = breakpoint" in self.res.GetOutput():
|
||||
breakpoint_stops += 1
|
||||
# Since there are only three worker threads that could hit the breakpoint.
|
||||
if breakpoint_stops > 3:
|
||||
self.fail("Do not expect to break more than 3 times")
|
||||
continue
|
||||
|
@ -96,6 +113,76 @@ class WatchpointForMultipleThreadsTestCase(TestBase):
|
|||
self.expect("watchpoint list -v",
|
||||
substrs = ['hit_count = 1'])
|
||||
|
||||
def hello_multiple_threads_wp_set_and_then_delete(self):
|
||||
"""Test that lldb watchpoint works for multiple threads, and after the watchpoint is deleted, the watchpoint event should no longer fires."""
|
||||
exe = os.path.join(os.getcwd(), self.exe_name)
|
||||
self.runCmd("file " + exe, CURRENT_EXECUTABLE_SET)
|
||||
|
||||
# Add a breakpoint to set a watchpoint when stopped on the breakpoint.
|
||||
self.expect("breakpoint set -l %d" % self.first_stop, BREAKPOINT_CREATED,
|
||||
startstr = "Breakpoint created: 1: file ='%s', line = %d, locations = 1" %
|
||||
(self.source, self.first_stop))
|
||||
|
||||
# Set this breakpoint to allow newly created thread to inherit the global watchpoint state.
|
||||
self.expect("breakpoint set -l %d" % self.thread_function, BREAKPOINT_CREATED,
|
||||
startstr = "Breakpoint created: 2: file ='%s', line = %d, locations = 1" %
|
||||
(self.source, self.thread_function))
|
||||
|
||||
# Run the program.
|
||||
self.runCmd("run", RUN_SUCCEEDED)
|
||||
|
||||
# We should be stopped again due to the breakpoint.
|
||||
# The stop reason of the thread should be breakpoint.
|
||||
self.expect("thread list", STOPPED_DUE_TO_BREAKPOINT,
|
||||
substrs = ['stopped',
|
||||
'stop reason = breakpoint'])
|
||||
|
||||
# Now let's set a write-type watchpoint for variable 'g_val'.
|
||||
# The main.cpp, by design, misbehaves by not following the agreed upon
|
||||
# protocol of using a mutex while accessing the global pool and by not
|
||||
# writing to the variable.
|
||||
self.expect("watchpoint set variable -w write g_val", WATCHPOINT_CREATED,
|
||||
substrs = ['Watchpoint created', 'size = 4', 'type = w'])
|
||||
|
||||
# Use the '-v' option to do verbose listing of the watchpoint.
|
||||
# The hit count should be 0 initially.
|
||||
self.expect("watchpoint list -v",
|
||||
substrs = ['hit_count = 0'])
|
||||
|
||||
breakpoint_stops = 0
|
||||
watchpoint_stops = 0
|
||||
while True:
|
||||
self.runCmd("process continue")
|
||||
self.runCmd("process status")
|
||||
if re.search("Process .* exited", self.res.GetOutput()):
|
||||
# Great, we are done with this test!
|
||||
break
|
||||
|
||||
self.runCmd("thread list")
|
||||
if "stop reason = breakpoint" in self.res.GetOutput():
|
||||
self.runCmd("thread backtrace all")
|
||||
breakpoint_stops += 1
|
||||
if self.TraceOn():
|
||||
print "breakpoint_stops=%d...." % breakpoint_stops
|
||||
# Since there are only three worker threads that could hit the breakpoint.
|
||||
if breakpoint_stops > 3:
|
||||
self.fail("Do not expect to break more than 3 times")
|
||||
continue
|
||||
elif "stop reason = watchpoint" in self.res.GetOutput():
|
||||
self.runCmd("thread backtrace all")
|
||||
watchpoint_stops += 1
|
||||
if watchpoint_stops > 1:
|
||||
self.fail("Watchpoint hits not supposed to exceed 1 by design!")
|
||||
# Good, we verified that the watchpoint works! Now delete the watchpoint.
|
||||
if self.TraceOn():
|
||||
print "watchpoint_stops=%d at the moment we delete the watchpoint" % watchpoint_stops
|
||||
self.runCmd("watchpoint delete 1")
|
||||
self.expect("watchpoint list -v",
|
||||
substrs = ['No watchpoints currently set.'])
|
||||
continue
|
||||
else:
|
||||
self.fail("The stop reason should be either break or watchpoint")
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
import atexit
|
||||
|
|
Loading…
Reference in New Issue