Skip to content

[benchmark] More StringBuilder/Interpolation benchmarks #14992

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merged
merged 2 commits into from
Mar 6, 2018
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
98 changes: 81 additions & 17 deletions benchmark/single-source/StringBuilder.swift
Original file line number Diff line number Diff line change
Expand Up @@ -13,10 +13,30 @@
import TestsUtils

public let StringBuilder = [
BenchmarkInfo(name: "StringAdder", runFunction: run_StringAdder, tags: [.validation, .api, .String]),
BenchmarkInfo(name: "StringBuilder", runFunction: run_StringBuilder, tags: [.validation, .api, .String]),
BenchmarkInfo(name: "StringBuilderLong", runFunction: run_StringBuilderLong, tags: [.validation, .api, .String]),
BenchmarkInfo(name: "StringUTF16Builder", runFunction: run_StringUTF16Builder, tags: [.validation, .api, .String]),
BenchmarkInfo(
name: "StringAdder",
runFunction: run_StringAdder,
tags: [.validation, .api, .String]),
BenchmarkInfo(
name: "StringBuilder",
runFunction: run_StringBuilder,
tags: [.validation, .api, .String]),
BenchmarkInfo(
name: "StringUTF16Builder",
runFunction: run_StringUTF16Builder,
tags: [.validation, .api, .String]),
BenchmarkInfo(
name: "StringUTF16SubstringBuilder",
runFunction: run_StringUTF16SubstringBuilder,
tags: [.validation, .api, .String]),
BenchmarkInfo(
name: "StringBuilderLong",
runFunction: run_StringBuilderLong,
tags: [.validation, .api, .String]),
BenchmarkInfo(
name: "StringBuilderWithLongSubstring",
runFunction: run_StringBuilderWithLongSubstring,
tags: [.validation, .api, .String]),
BenchmarkInfo(
name: "StringWordBuilder",
runFunction: run_StringWordBuilder,
Expand All @@ -29,7 +49,7 @@ public let StringBuilder = [

@inline(never)
func buildString(_ i: String) -> String {
var sb = i
var sb = getString(i)
for str in ["b","c","d","pizza"] {
sb += str
}
Expand All @@ -39,54 +59,95 @@ func buildString(_ i: String) -> String {
@inline(never)
public func run_StringBuilder(_ N: Int) {
for _ in 1...5000*N {
_ = buildString(getString("a"))
blackHole(buildString("a"))
}
}

@inline(never)
func addString(_ i: String) -> String {
let s = i + "b" + "c" + "d" + "pizza"
let s = getString(i) + "b" + "c" + "d" + "pizza"
return s
}

@inline(never)
public func run_StringAdder(_ N: Int) {
for _ in 1...5000*N {
_ = addString(getString("a"))
blackHole(addString("a"))
}
}

@inline(never)
func buildStringUTF16(_ i: String) -> String {
var sb = i
var sb = getString(i)
for str in ["🎉","c","d","pizza"] {
sb += str
}
return sb
}

@inline(never)
func buildStringFromSmallSubstrings(_ i: String) -> String {
var sb = getString(i)
for str in ["_🎉","cd","de","pizza"] {
sb += str.dropFirst()
}
return sb
}

@inline(never)
public func run_StringUTF16Builder(_ N: Int) {
for _ in 1...5000*N {
_ = buildStringUTF16("a")
blackHole(buildStringUTF16("a"))
}
}

@inline(never)
public func run_StringUTF16SubstringBuilder(_ N: Int) {
for _ in 1...5000*N {
blackHole(buildStringFromSmallSubstrings("a"))
}
}

func getLongString() -> String {
let long = """
Swift is a multi-paradigm, compiled programming language created for
iOS, OS X, watchOS, tvOS and Linux development by Apple Inc. Swift is
designed to work with Apple's Cocoa and Cocoa Touch frameworks and the
large body of existing Objective-C code written for Apple products. Swift
is intended to be more resilient to erroneous code (\"safer\") than
Objective-C and also more concise. It is built with the LLVM compiler
framework included in Xcode 6 and later and uses the Objective-C runtime,
which allows C, Objective-C, C++ and Swift code to run within a single
program.
"""
return getString(long)
}

@inline(never)
func buildStringLong(_ i: String) -> String {
var sb = i
let long = "Swift is a multi-paradigm, compiled programming language created for iOS, OS X, watchOS, tvOS and Linux development by Apple Inc. Swift is designed to work with Apple's Cocoa and Cocoa Touch frameworks and the large body of existing Objective-C code written for Apple products. Swift is intended to be more resilient to erroneous code (\"safer\") than Objective-C and also more concise. It is built with the LLVM compiler framework included in Xcode 6 and later and uses the Objective-C runtime, which allows C, Objective-C, C++ and Swift code to run within a single program."
sb += long
var sb = getString(i)
sb += getLongString()
return sb
}


@inline(never)
func buildStringWithLongSubstring(_ i: String) -> String {
var sb = getString(i)
sb += getLongString().dropFirst()
return sb
}

@inline(never)
public func run_StringBuilderLong(_ N: Int) {
for _ in 1...5000*N {
_ = buildStringLong("👻")
blackHole(buildStringLong("👻"))
}
}

@inline(never)
public func run_StringBuilderWithLongSubstring(_ N: Int) {
for _ in 1...5000*N {
blackHole(buildStringWithLongSubstring("👻"))
}
}

Expand All @@ -96,6 +157,7 @@ func buildString(
count: Int,
reservingCapacity: Bool
) -> String {
let word = getString(word)
var sb = ""
if reservingCapacity {
sb.reserveCapacity(count * word.unicodeScalars.count)
Expand All @@ -108,10 +170,12 @@ func buildString(

@inline(never)
public func run_StringWordBuilder(_ N: Int) {
_ = buildString(word: "bumfuzzle", count: 50_000 * N, reservingCapacity: false)
blackHole(buildString(
word: "bumfuzzle", count: 50_000 * N, reservingCapacity: false))
}

@inline(never)
public func run_StringWordBuilderReservingCapacity(_ N: Int) {
_ = buildString(word: "bumfuzzle", count: 50_000 * N, reservingCapacity: true)
blackHole(buildString(
word: "bumfuzzle", count: 50_000 * N, reservingCapacity: true))
}
63 changes: 61 additions & 2 deletions benchmark/single-source/StringInterpolation.swift
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,14 @@ public let StringInterpolation = BenchmarkInfo(
name: "StringInterpolation",
runFunction: run_StringInterpolation,
tags: [.validation, .api, .String])
public let StringInterpolationSmall = BenchmarkInfo(
name: "StringInterpolationSmall",
runFunction: run_StringInterpolationSmall,
tags: [.validation, .api, .String])
public let StringInterpolationManySmallSegments = BenchmarkInfo(
name: "StringInterpolationManySmallSegments",
runFunction: run_StringInterpolationManySmallSegments,
tags: [.validation, .api, .String])

class RefTypePrintable : CustomStringConvertible {
var description: String {
Expand All @@ -33,15 +41,66 @@ public func run_StringInterpolation(_ N: Int) {
for _ in 1...100*N {
var result = 0
for _ in 1...reps {
let s = "\(anInt) abcdefdhijklmn \(aRefCountedObject) abcdefdhijklmn \u{01}"
let s: String = getString(
"\(anInt) abcdefdhijklmn \(aRefCountedObject) abcdefdhijklmn \u{01}")
let utf16 = s.utf16

// FIXME: if String is not stored as UTF-16 on this platform, then the
// following operation has a non-trivial cost and needs to be replaced
// with an operation on the native storage type.
result = result &+ Int(utf16[utf16.index(before: utf16.endIndex)])
result = result &+ Int(utf16.last!)
blackHole(s)
}
CheckResults(result == refResult)
}
}

@inline(never)
public func run_StringInterpolationSmall(_ N: Int) {
let reps = 100
let refResult = reps
let anInt: Int64 = 0x42

for _ in 1...100*N {
var result = 0
for _ in 1...reps {
let s: String = getString(
"\(getString("int")): \(anInt) \(getString("abc")) \u{01}")
result = result &+ Int(s.utf8.last!)
blackHole(s)
}
CheckResults(result == refResult)
}
}

@inline(never)
public func run_StringInterpolationManySmallSegments(_ N: Int) {
let numHex = min(UInt64(N), 0x0FFF_FFFF_FFFF_FFFF)
let numOct = min(UInt64(N), 0x0000_03FF_FFFF_FFFF)
let numBin = min(UInt64(N), 0x7FFF)
let segments = [
"abc",
String(numHex, radix: 16),
"0123456789",
String(Double.pi/2),
"*barely* small!",
String(numOct, radix: 8),
"",
String(numBin, radix: 2),
]
assert(segments.count == 8)

func getSegment(_ i: Int) -> String {
return getString(segments[i])
}

let reps = 100
for _ in 1...100*N {
for _ in 1...reps {
blackHole("""
\(getSegment(0)) \(getSegment(1))/\(getSegment(2))_\(getSegment(3))
\(getSegment(4)) \(getSegment(5)), \(getSegment(6))~~\(getSegment(7))
""")
}
}
}
2 changes: 2 additions & 0 deletions benchmark/utils/main.swift
Original file line number Diff line number Diff line change
Expand Up @@ -276,6 +276,8 @@ registerBenchmark(StringComparison)
registerBenchmark(StringEdits)
registerBenchmark(StringEnum)
registerBenchmark(StringInterpolation)
registerBenchmark(StringInterpolationSmall)
registerBenchmark(StringInterpolationManySmallSegments)
registerBenchmark(StringMatch)
registerBenchmark(StringRemoveDupes)
registerBenchmark(StringTests)
Expand Down