CompletionQueue.swift 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. /*
  2. * Copyright 2016, gRPC Authors All rights reserved.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. #if SWIFT_PACKAGE
  17. import CgRPC
  18. import Dispatch
  19. #endif
  20. import Foundation
  21. /// A type indicating the kind of event returned by the completion queue
  22. enum CompletionType {
  23. case queueShutdown
  24. case queueTimeout
  25. case complete
  26. case unknown
  27. fileprivate static func completionType(_ value: grpc_completion_type) -> CompletionType {
  28. switch value {
  29. case GRPC_QUEUE_SHUTDOWN:
  30. return .queueShutdown
  31. case GRPC_QUEUE_TIMEOUT:
  32. return .queueTimeout
  33. case GRPC_OP_COMPLETE:
  34. return .complete
  35. default:
  36. return .unknown
  37. }
  38. }
  39. }
  40. /// An event that is returned by the completion queue
  41. struct CompletionQueueEvent {
  42. let type: CompletionType
  43. let success: Int32
  44. let tag: Int64
  45. init(_ event: grpc_event) {
  46. type = CompletionType.completionType(event.type)
  47. success = event.success
  48. tag = cgrpc_event_tag(event)
  49. }
  50. }
  51. /// A gRPC Completion Queue
  52. class CompletionQueue {
  53. /// Optional user-provided name for the queue
  54. let name: String?
  55. /// Pointer to underlying C representation
  56. private let underlyingCompletionQueue: UnsafeMutableRawPointer
  57. /// Operation groups that are awaiting completion, keyed by tag
  58. private var operationGroups: [Int64: OperationGroup] = [:]
  59. /// Mutex for synchronizing access to operationGroups
  60. private let operationGroupsMutex: Mutex = Mutex()
  61. private var hasBeenShutdown = false
  62. /// Initializes a CompletionQueue
  63. ///
  64. /// - Parameter cq: the underlying C representation
  65. init(underlyingCompletionQueue: UnsafeMutableRawPointer, name: String? = nil) {
  66. // The underlying completion queue is NOT OWNED by this class, so we don't dealloc it in a deinit
  67. self.underlyingCompletionQueue = underlyingCompletionQueue
  68. self.name = name
  69. }
  70. deinit {
  71. operationGroupsMutex.synchronize {
  72. hasBeenShutdown = true
  73. }
  74. cgrpc_completion_queue_shutdown(underlyingCompletionQueue)
  75. cgrpc_completion_queue_drain(underlyingCompletionQueue)
  76. grpc_completion_queue_destroy(underlyingCompletionQueue)
  77. }
  78. /// Waits for an operation group to complete
  79. ///
  80. /// - Parameter timeout: a timeout value in seconds
  81. /// - Returns: a grpc_completion_type code indicating the result of waiting
  82. func wait(timeout: TimeInterval) -> CompletionQueueEvent {
  83. let event = cgrpc_completion_queue_get_next_event(underlyingCompletionQueue, timeout)
  84. return CompletionQueueEvent(event)
  85. }
  86. /// Register an operation group for handling upon completion. Returns true if the operation group was registered
  87. /// successfully.
  88. ///
  89. /// - Parameter operationGroup: the operation group to handle
  90. func register(_ operationGroup: OperationGroup, onSuccess: () throws -> Void) rethrows {
  91. try operationGroupsMutex.synchronize {
  92. guard !hasBeenShutdown
  93. else { throw CallError.completionQueueShutdown }
  94. operationGroups[operationGroup.tag] = operationGroup
  95. try onSuccess()
  96. }
  97. }
  98. /// Runs a completion queue and call a completion handler when finished
  99. ///
  100. /// - Parameter completion: a completion handler that is called when the queue stops running
  101. func runToCompletion(completion: (() -> Void)?) {
  102. // run the completion queue on a new background thread
  103. DispatchQueue.global().async {
  104. spinloop: while true {
  105. let event = cgrpc_completion_queue_get_next_event(self.underlyingCompletionQueue, 600)
  106. switch event.type {
  107. case GRPC_OP_COMPLETE:
  108. let tag = cgrpc_event_tag(event)
  109. self.operationGroupsMutex.lock()
  110. let operationGroup = self.operationGroups[tag]
  111. self.operationGroupsMutex.unlock()
  112. if let operationGroup = operationGroup {
  113. // call the operation group completion handler
  114. operationGroup.success = (event.success == 1)
  115. operationGroup.completion?(operationGroup)
  116. self.operationGroupsMutex.synchronize {
  117. self.operationGroups[tag] = nil
  118. }
  119. } else {
  120. print("CompletionQueue.runToCompletion error: operation group with tag \(tag) not found")
  121. }
  122. case GRPC_QUEUE_SHUTDOWN:
  123. self.operationGroupsMutex.lock()
  124. let currentOperationGroups = self.operationGroups
  125. self.operationGroups = [:]
  126. self.operationGroupsMutex.unlock()
  127. for operationGroup in currentOperationGroups.values {
  128. operationGroup.success = false
  129. operationGroup.completion?(operationGroup)
  130. }
  131. break spinloop
  132. case GRPC_QUEUE_TIMEOUT:
  133. continue spinloop
  134. default:
  135. print("CompletionQueue.runToCompletion error: unknown event type \(event.type)")
  136. break spinloop
  137. }
  138. }
  139. // when the queue stops running, call the queue completion handler
  140. completion?()
  141. }
  142. }
  143. /// Runs a completion queue
  144. func run() {
  145. runToCompletion(completion: nil)
  146. }
  147. /// Shuts down a completion queue
  148. func shutdown() {
  149. operationGroupsMutex.synchronize {
  150. hasBeenShutdown = true
  151. }
  152. cgrpc_completion_queue_shutdown(underlyingCompletionQueue)
  153. }
  154. }