Skip to content

Commit

Permalink
Add thread safety to Clock
Browse files Browse the repository at this point in the history
Signed-off-by: Seweryn Plażuk <[email protected]>
  • Loading branch information
Seweryn Plażuk authored and sewerynplazuk committed May 22, 2024
1 parent 1b2d25b commit 22cb9c9
Show file tree
Hide file tree
Showing 5 changed files with 49 additions and 4 deletions.
4 changes: 4 additions & 0 deletions Kronos.xcodeproj/project.pbxproj
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@
930B39DD2051E6D300360BA2 /* TimeStorage.swift in Sources */ = {isa = PBXBuildFile; fileRef = 930B39DC2051E6D300360BA2 /* TimeStorage.swift */; };
930B39E02051F26500360BA2 /* TimeStorageTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 930B39DE2051F25300360BA2 /* TimeStorageTests.swift */; };
C201748E1BD5509D00E4FE18 /* Kronos.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = C20174831BD5509D00E4FE18 /* Kronos.framework */; };
E5E7A7702BFE559F0025D9D0 /* UnfairLock.swift in Sources */ = {isa = PBXBuildFile; fileRef = E5E7A76F2BFE559F0025D9D0 /* UnfairLock.swift */; };
/* End PBXBuildFile section */

/* Begin PBXContainerItemProxy section */
Expand Down Expand Up @@ -58,6 +59,7 @@
C2C036D41C2B180D003FB853 /* UniversalFramework_Base.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; path = UniversalFramework_Base.xcconfig; sourceTree = "<group>"; };
C2C036D51C2B180D003FB853 /* UniversalFramework_Framework.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; path = UniversalFramework_Framework.xcconfig; sourceTree = "<group>"; };
C2C036D61C2B180D003FB853 /* UniversalFramework_Test.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; path = UniversalFramework_Test.xcconfig; sourceTree = "<group>"; };
E5E7A76F2BFE559F0025D9D0 /* UnfairLock.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UnfairLock.swift; sourceTree = "<group>"; };
/* End PBXFileReference section */

/* Begin PBXFrameworksBuildPhase section */
Expand Down Expand Up @@ -124,6 +126,7 @@
26447D7B1D6E54D400159BEE /* TimeFreeze.swift */,
930B39DC2051E6D300360BA2 /* TimeStorage.swift */,
5DB5A05F2BAAF67D0069CCF9 /* PrivacyInfo.xcprivacy */,
E5E7A76F2BFE559F0025D9D0 /* UnfairLock.swift */,
);
path = Sources;
sourceTree = "<group>";
Expand Down Expand Up @@ -234,6 +237,7 @@
26447D7E1D6E54D400159BEE /* InternetAddress.swift in Sources */,
26447D811D6E54D400159BEE /* NTPClient.swift in Sources */,
26447D7F1D6E54D400159BEE /* Data+Bytes.swift in Sources */,
E5E7A7702BFE559F0025D9D0 /* UnfairLock.swift in Sources */,
26447D841D6E54D400159BEE /* TimeFreeze.swift in Sources */,
26447D821D6E54D400159BEE /* NTPPacket.swift in Sources */,
26447D801D6E54D400159BEE /* NSTimer+ClosureKit.swift in Sources */,
Expand Down
17 changes: 14 additions & 3 deletions Sources/Clock.swift
Original file line number Diff line number Diff line change
Expand Up @@ -25,9 +25,20 @@ public typealias AnnotatedTime = (
/// print(Clock.now)
/// ```
public struct Clock {
private static var stableTime: TimeFreeze? {
didSet {
self.storage.stableTime = self.stableTime
private static let lock = UnfairLock()

private static var _stableTime: TimeFreeze?
static var stableTime: TimeFreeze? {
get {
self.lock.around {
return self._stableTime
}
}
set {
self.lock.around {
self._stableTime = newValue
self.storage.stableTime = newValue
}
}
}

Expand Down
2 changes: 1 addition & 1 deletion Sources/TimeFreeze.swift
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ private let kOffsetKey = "Offset"
struct TimeFreeze {
private let uptime: TimeInterval
private let timestamp: TimeInterval
private let offset: TimeInterval
let offset: TimeInterval

/// The stable timestamp adjusted by the most accurate offset known so far.
var adjustedTimestamp: TimeInterval {
Expand Down
29 changes: 29 additions & 0 deletions Sources/UnfairLock.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
import Foundation

final class UnfairLock {
private let unfairLock: os_unfair_lock_t

init() {
unfairLock = .allocate(capacity: 1)
unfairLock.initialize(to: os_unfair_lock())
}

deinit {
unfairLock.deinitialize(count: 1)
unfairLock.deallocate()
}

func around<T>(_ block: () -> T) -> T {
lock()
defer { unlock() }
return block()
}

private func lock() {
os_unfair_lock_lock(unfairLock)
}

private func unlock() {
os_unfair_lock_unlock(unfairLock)
}
}
1 change: 1 addition & 0 deletions Tests/KronosTests/ClockTests.swift
Original file line number Diff line number Diff line change
Expand Up @@ -38,4 +38,5 @@ final class ClockTests: XCTestCase {

self.waitForExpectations(timeout: 20)
}

}

0 comments on commit 22cb9c9

Please sign in to comment.