-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathConvTest2.swift
130 lines (113 loc) · 3.57 KB
/
ConvTest2.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
//
// ConvTest2.swift
// Modification of ConvTest, to execute a fixed amount of work, and use dispatch
// groups to wait for work to complete, rather than dispatch_main.
// This was primarily while debugging the 'Linux process becomes a zombie' issue
// with dispatch_main.
// See: https://github.com/apple/swift-corelibs-libdispatch/pull/99
//
// Created by David Jones (@djones6) on 08/06/2016
//
import Foundation
import Dispatch
// Determine how many concurrent blocks to schedule (user specified, or 10)
var CONCURRENCY:Int = 10
// Determines how many times to convert a string per block
var EFFORT:Int = 1000
// Determines the length of the payload being converted
var LENGTH:Int = 1000
// Determines how many times each block should be dispatched before terminating
var NUM_LOOPS:Int = 100
// Debug
var DEBUG = false
func usage() {
print("Options are:")
print(" -c, --concurrency n: number of concurrent Dispatch blocks")
print(" -e, --effort n: number of times to invoke conversion per block")
print(" -l, --length n: length of String (in chars) to be converted")
print(" -d, --debug: print a lot of debugging output")
exit(1)
}
// Parse an expected int value provided on the command line
func parseInt(param: String, value: String) -> Int {
if let userInput = Int(value) {
return userInput
} else {
print("Invalid value for \(param): '\(value)'")
exit(1)
}
}
// Parse command line options
var param:String? = nil
var remainingArgs = Process.arguments.dropFirst(1)
for arg in remainingArgs {
if let _param = param {
param = nil
switch _param {
case "-c", "--concurrency":
CONCURRENCY = parseInt(param: _param, value: arg)
case "-e", "--effort":
EFFORT = parseInt(param: _param, value: arg)
case "-l", "--length":
LENGTH = parseInt(param: _param, value: arg)
default:
print("Invalid option '\(arg)'")
usage()
}
} else {
switch arg {
case "-c", "--concurrency", "-e", "--effort", "-l", "--length":
param = arg
case "-d", "--debug":
DEBUG = true
case "-?", "-h", "--help", "--?":
usage()
default:
print("Invalid option '\(arg)'")
usage()
}
}
}
if (DEBUG) {
print("Concurrency: \(CONCURRENCY)")
print("Effort: \(EFFORT)")
print("Length: \(LENGTH)")
print("Debug: \(DEBUG)")
}
// The string to convert
let PAYLOAD:String
var _payload = "Houston we have a problem"
while _payload.characters.count < LENGTH {
_payload = _payload + _payload
}
// Surely this isn't the best way to substring? but it works...
PAYLOAD = String(_payload.characters.dropLast(_payload.characters.count - LENGTH))
if DEBUG { print("Payload is \(PAYLOAD.characters.count) chars") }
// Create a queue to run blocks in parallel
let queue = dispatch_queue_create("hello", DISPATCH_QUEUE_CONCURRENT)
let group = dispatch_group_create()
// Block to be scheduled
func code(block: Int, loops: Int) -> () -> Void {
return {
for _ in 1...EFFORT {
let _ = PAYLOAD.data(using: NSUTF8StringEncoding)
}
if DEBUG { print("Instance \(block) done") }
// Dispatch a new block to replace this one
if (loops < NUM_LOOPS) {
dispatch_group_async(group, queue, code(block: block, loops: loops+1))
} else {
print("Block \(block) completed \(loops) loops")
}
}
}
print("Queueing \(CONCURRENCY) blocks")
// Queue the initial blocks
for i in 1...CONCURRENCY {
dispatch_group_async(group, queue, code(block: i, loops: 0))
}
print("Go!")
// Go
//dispatch_main()
dispatch_group_wait(group, dispatch_time(DISPATCH_TIME_NOW, 5000000000)) // 5 seconds
print("Waited 5 seconds, or completed")