The previous patch was reverted because the test fails to build when libsanitizers is not present. This patch catches the BuildError exception and skips the test appropriately. This patch tests LLDB integration with libsanitizers for ASan. rdar://111856681
210 lines
6.9 KiB
Python
210 lines
6.9 KiB
Python
"""
|
|
Test that ASan memory history provider returns correct stack traces
|
|
"""
|
|
|
|
|
|
import lldb
|
|
from lldbsuite.test.decorators import *
|
|
from lldbsuite.test.lldbtest import *
|
|
from lldbsuite.test import lldbplatform
|
|
from lldbsuite.test import lldbutil
|
|
from lldbsuite.test_event.build_exception import BuildError
|
|
|
|
class AsanTestCase(TestBase):
|
|
@skipIfFreeBSD # llvm.org/pr21136 runtimes not yet available by default
|
|
@expectedFailureNetBSD
|
|
@skipUnlessAddressSanitizer
|
|
def test(self):
|
|
self.build(make_targets=["asan"])
|
|
self.asan_tests()
|
|
|
|
@skipIf(oslist=no_match(["macosx"]))
|
|
def test_libsanitizers_asan(self):
|
|
try:
|
|
self.build(make_targets=["libsanitizers"])
|
|
except BuildError as e:
|
|
self.skipTest("failed to build with libsanitizers")
|
|
self.libsanitizer_tests()
|
|
|
|
def setUp(self):
|
|
# Call super's setUp().
|
|
TestBase.setUp(self)
|
|
self.line_malloc = line_number("main.c", "// malloc line")
|
|
self.line_malloc2 = line_number("main.c", "// malloc2 line")
|
|
self.line_free = line_number("main.c", "// free line")
|
|
self.line_breakpoint = line_number("main.c", "// break line")
|
|
|
|
# Test line numbers: rdar://126237493
|
|
def libsanitizer_tests(self):
|
|
target = self.createTestTarget()
|
|
|
|
self.runCmd(
|
|
"env SanitizersAddress=1 MallocSanitizerZone=1 MallocSecureAllocator=0"
|
|
)
|
|
|
|
self.runCmd("run")
|
|
|
|
# In libsanitizers, memory history is not supported until a report has been generated
|
|
self.expect(
|
|
"thread list",
|
|
"Process should be stopped due to ASan report",
|
|
substrs=["stopped", "stop reason = Use of deallocated memory"],
|
|
)
|
|
|
|
# test the 'memory history' command
|
|
self.expect(
|
|
"memory history 'pointer'",
|
|
substrs=[
|
|
"Memory deallocated by Thread",
|
|
"a.out`f2",
|
|
"main.c",
|
|
"Memory allocated by Thread",
|
|
"a.out`f1",
|
|
"main.c",
|
|
],
|
|
)
|
|
|
|
# do the same using SB API
|
|
process = self.dbg.GetSelectedTarget().process
|
|
val = (
|
|
process.GetSelectedThread().GetSelectedFrame().EvaluateExpression("pointer")
|
|
)
|
|
addr = val.GetValueAsUnsigned()
|
|
threads = process.GetHistoryThreads(addr)
|
|
self.assertEqual(threads.GetSize(), 2)
|
|
|
|
history_thread = threads.GetThreadAtIndex(0)
|
|
self.assertTrue(history_thread.num_frames >= 2)
|
|
self.assertEqual(
|
|
history_thread.frames[1].GetLineEntry().GetFileSpec().GetFilename(),
|
|
"main.c",
|
|
)
|
|
|
|
history_thread = threads.GetThreadAtIndex(1)
|
|
self.assertTrue(history_thread.num_frames >= 2)
|
|
self.assertEqual(
|
|
history_thread.frames[1].GetLineEntry().GetFileSpec().GetFilename(),
|
|
"main.c",
|
|
)
|
|
|
|
# let's free the container (SBThreadCollection) and see if the
|
|
# SBThreads still live
|
|
threads = None
|
|
self.assertTrue(history_thread.num_frames >= 2)
|
|
self.assertEqual(
|
|
history_thread.frames[1].GetLineEntry().GetFileSpec().GetFilename(),
|
|
"main.c",
|
|
)
|
|
|
|
def asan_tests(self):
|
|
target = self.createTestTarget()
|
|
|
|
self.registerSanitizerLibrariesWithTarget(target)
|
|
|
|
self.runCmd("breakpoint set -f main.c -l %d" % self.line_breakpoint)
|
|
|
|
# "memory history" command should not work without a process
|
|
self.expect(
|
|
"memory history 0",
|
|
error=True,
|
|
substrs=["Command requires a current process"],
|
|
)
|
|
|
|
self.runCmd("run")
|
|
|
|
stop_reason = (
|
|
self.dbg.GetSelectedTarget().process.GetSelectedThread().GetStopReason()
|
|
)
|
|
if stop_reason == lldb.eStopReasonExec:
|
|
# On OS X 10.10 and older, we need to re-exec to enable
|
|
# interceptors.
|
|
self.runCmd("continue")
|
|
|
|
# the stop reason of the thread should be breakpoint.
|
|
self.expect(
|
|
"thread list",
|
|
STOPPED_DUE_TO_BREAKPOINT,
|
|
substrs=["stopped", "stop reason = breakpoint"],
|
|
)
|
|
|
|
# test that the ASan dylib is present
|
|
self.expect(
|
|
"image lookup -n __asan_describe_address",
|
|
"__asan_describe_address should be present",
|
|
substrs=["1 match found"],
|
|
)
|
|
|
|
# test the 'memory history' command
|
|
self.expect(
|
|
"memory history 'pointer'",
|
|
substrs=[
|
|
"Memory deallocated by Thread",
|
|
"a.out`f2",
|
|
"main.c:%d" % self.line_free,
|
|
"Memory allocated by Thread",
|
|
"a.out`f1",
|
|
"main.c:%d" % self.line_malloc,
|
|
],
|
|
)
|
|
|
|
# do the same using SB API
|
|
process = self.dbg.GetSelectedTarget().process
|
|
val = (
|
|
process.GetSelectedThread().GetSelectedFrame().EvaluateExpression("pointer")
|
|
)
|
|
addr = val.GetValueAsUnsigned()
|
|
threads = process.GetHistoryThreads(addr)
|
|
self.assertEqual(threads.GetSize(), 2)
|
|
|
|
history_thread = threads.GetThreadAtIndex(0)
|
|
self.assertGreaterEqual(history_thread.num_frames, 2)
|
|
self.assertEqual(
|
|
history_thread.frames[1].GetLineEntry().GetFileSpec().GetFilename(),
|
|
"main.c",
|
|
)
|
|
self.assertEqual(
|
|
history_thread.frames[1].GetLineEntry().GetLine(), self.line_free
|
|
)
|
|
|
|
history_thread = threads.GetThreadAtIndex(1)
|
|
self.assertGreaterEqual(history_thread.num_frames, 2)
|
|
self.assertEqual(
|
|
history_thread.frames[1].GetLineEntry().GetFileSpec().GetFilename(),
|
|
"main.c",
|
|
)
|
|
self.assertEqual(
|
|
history_thread.frames[1].GetLineEntry().GetLine(), self.line_malloc
|
|
)
|
|
|
|
# let's free the container (SBThreadCollection) and see if the
|
|
# SBThreads still live
|
|
threads = None
|
|
self.assertGreaterEqual(history_thread.num_frames, 2)
|
|
self.assertEqual(
|
|
history_thread.frames[1].GetLineEntry().GetFileSpec().GetFilename(),
|
|
"main.c",
|
|
)
|
|
self.assertEqual(
|
|
history_thread.frames[1].GetLineEntry().GetLine(), self.line_malloc
|
|
)
|
|
|
|
# ASan will break when a report occurs and we'll try the API then
|
|
self.runCmd("continue")
|
|
|
|
self.expect(
|
|
"thread list",
|
|
"Process should be stopped due to ASan report",
|
|
substrs=["stopped", "stop reason = Use of deallocated memory"],
|
|
)
|
|
|
|
# make sure the 'memory history' command still works even when we're
|
|
# generating a report now
|
|
self.expect(
|
|
"memory history 'another_pointer'",
|
|
substrs=[
|
|
"Memory allocated by Thread",
|
|
"a.out`f1",
|
|
"main.c:%d" % self.line_malloc2,
|
|
],
|
|
)
|