mirror of
https://github.com/apple/swift.git
synced 2025-12-25 12:15:36 +01:00
The initial version of the debugger testing transform instruments
assignments in a way that allows the debugger to sanity-check its
expression evaluator.
Given an assignment expression of the form:
```
a = b
```
The transform rewrites the relevant bits of the AST to look like this:
```
{ () -> () in
a = b
checkExpect("a", stringForPrintObject(a))
}()
```
The purpose of the rewrite is to make it easier to exercise the
debugger's expression evaluator in new contexts. This can be automated
by having the debugger set a breakpoint on checkExpect, running `expr
$Varname`, and comparing the result to the expected value generated by
the runtime.
While the initial version of this testing transform only supports
instrumenting assignments, it should be simple to teach it to do more
interesting rewrites.
There's a driver script available in SWIFT_BIN_DIR/lldb-check-expect to
simplfiy the process of launching and testing instrumented programs.
rdar://36032055
117 lines
3.8 KiB
Swift
117 lines
3.8 KiB
Swift
// RUN: %target-run-simple-swift
|
|
// REQUIRES: executable_test, objc_interop
|
|
|
|
import Foundation
|
|
import StdlibUnittest
|
|
|
|
let StringForPrintObjectTests = TestSuite("StringForPrintObject")
|
|
|
|
// Wrap stringForPrintObject for convenience. Note that the debugger uses
|
|
// something slightly different to pretty-print (see: debugVal()).
|
|
func printObj<T>(_ x: T) -> String {
|
|
return _stringForPrintObject(x)
|
|
}
|
|
|
|
// Check if @x has a reference type.
|
|
func hasReferenceType<T>(_ x: T) -> Bool {
|
|
return _canBeClass(T.self) == 1
|
|
}
|
|
|
|
// The debugger uses unsafeBitCast to take an arbitrary address and cast it to
|
|
// AnyObject. Mimic that operation here.
|
|
func debugVal<T>(_ x: inout T) -> String {
|
|
if !hasReferenceType(x) {
|
|
return printObj(x)
|
|
}
|
|
return withUnsafePointer(to: &x) {
|
|
return _stringForPrintObject(Swift.unsafeBitCast($0.pointee, to: AnyObject.self))
|
|
}
|
|
}
|
|
|
|
// Check if @x uses the small-string or Cocoa representations.
|
|
func hasSmallStringOrCocoaVariant(_ x: String) -> Bool {
|
|
return x._guts._isCocoa || x._guts._isSmall
|
|
}
|
|
|
|
StringForPrintObjectTests.test("Basic") {
|
|
var a = "Hello World" as NSString
|
|
let a_printed = printObj(a)
|
|
let a_debug = debugVal(&a)
|
|
expectEqual("Hello World", String(reflecting: a))
|
|
expectEqual("Hello World\n", a_printed)
|
|
expectEqual(a_printed, a_debug)
|
|
}
|
|
|
|
StringForPrintObjectTests.test("NSStringFromStringLiteral") {
|
|
var a = Foundation.NSString(stringLiteral: "Hello World")
|
|
let a_printed = printObj(a)
|
|
let a_debug = debugVal(&a)
|
|
expectEqual("Hello World", String(reflecting: a))
|
|
expectEqual("Hello World\n", a_printed)
|
|
expectEqual(a_printed, a_debug)
|
|
}
|
|
|
|
StringForPrintObjectTests.test("NSStringFromUnsafeBuffer") {
|
|
let buf = UnsafeMutablePointer<Int8>.allocate(capacity: 8)
|
|
buf[0] = 65
|
|
buf[1] = 0
|
|
var a = Foundation.NSString(utf8String: buf)!
|
|
let a_printed = printObj(a)
|
|
let a_debug = debugVal(&a)
|
|
expectEqual("A", String(reflecting: a))
|
|
expectEqual("A\n", a_printed)
|
|
expectEqual(a_printed, a_debug)
|
|
buf.deallocate()
|
|
}
|
|
|
|
StringForPrintObjectTests.test("NSStringUTF8") {
|
|
let nsUTF16 = NSString(utf8String: "🏂☃❅❆❄︎⛄️❄️")!
|
|
expectTrue(CFStringGetCharactersPtr(unsafeBitCast(nsUTF16, to: CFString.self)) != nil)
|
|
var newNSUTF16 = nsUTF16 as String
|
|
expectTrue(hasSmallStringOrCocoaVariant(newNSUTF16))
|
|
let printed = printObj(newNSUTF16)
|
|
let debug = debugVal(&newNSUTF16)
|
|
expectEqual("🏂☃❅❆❄︎⛄️❄️", String(reflecting: nsUTF16))
|
|
expectEqual("\"🏂☃❅❆❄︎⛄️❄️\"", String(reflecting: newNSUTF16))
|
|
expectEqual("\"🏂☃❅❆❄︎⛄️❄️\"\n", printed)
|
|
expectEqual(printed, debug)
|
|
}
|
|
|
|
StringForPrintObjectTests.test("ArrayOfStrings") {
|
|
var a = ["Hello World" as NSString]
|
|
let a_printed = printObj(a)
|
|
let a_debug = debugVal(&a)
|
|
expectEqual("[Hello World]", String(reflecting: a))
|
|
expectEqual("▿ 1 element\n - 0 : Hello World\n", a_printed)
|
|
expectEqual(a_printed, a_debug)
|
|
}
|
|
|
|
struct StructWithOneMember {
|
|
var a = "Hello World" as NSString
|
|
}
|
|
|
|
StringForPrintObjectTests.test("StructWithOneMember") {
|
|
var a = StructWithOneMember()
|
|
let a_printed = printObj(StructWithOneMember())
|
|
let a_debug = debugVal(&a)
|
|
expectEqual("main.StructWithOneMember(a: Hello World)", String(reflecting: a))
|
|
expectEqual("▿ StructWithOneMember\n - a : Hello World\n", a_printed)
|
|
expectEqual(a_printed, a_debug)
|
|
}
|
|
|
|
struct StructWithTwoMembers {
|
|
var a = 1
|
|
var b = "Hello World" as NSString
|
|
}
|
|
|
|
StringForPrintObjectTests.test("StructWithTwoMembers") {
|
|
var a = StructWithTwoMembers()
|
|
let a_printed = printObj(StructWithTwoMembers())
|
|
let a_debug = debugVal(&a)
|
|
expectEqual("main.StructWithTwoMembers(a: 1, b: Hello World)", String(reflecting: a))
|
|
expectEqual("▿ StructWithTwoMembers\n - a : 1\n - b : Hello World\n", a_printed)
|
|
expectEqual(a_printed, a_debug)
|
|
}
|
|
|
|
runAllTests()
|