Skip to content

Commit 8aa161b

Browse files
authored
Rollup merge of rust-lang#59374 - faern:simplify-checked-duration-since, r=shepmaster
Simplify checked_duration_since This follows the same design as we updated to in rust-lang#56490. Internally, all the system specific time implementations are checked, no panics. Then the panicking publicly exported API can just call the checked version of itself and make do with a single panic (`expect`) at the top. Since the internal sys implementations are now checked, this gets rid of the extra `if self >= &earlier` check in `checked_duration_since`. Except likely making the generated machine code simpler, it also reduces the algorithm from "Check panic condition -> call possibly panicking method" to just "call non panicking method". Added two test cases: * Edge case: Make sure `checked_duration_since` on two equal `Instant`s produce a zero duration, not a `None`. * Most common/intended usage: Make sure `later.checked_duration_since(earlier)`, returns an expected value.
2 parents cb15da7 + 1ccad16 commit 8aa161b

File tree

7 files changed

+29
-37
lines changed

7 files changed

+29
-37
lines changed

src/libstd/sys/cloudabi/time.rs

+3-5
Original file line numberDiff line numberDiff line change
@@ -33,11 +33,9 @@ impl Instant {
3333
Instant { t: 0 }
3434
}
3535

36-
pub fn sub_instant(&self, other: &Instant) -> Duration {
37-
let diff = self.t
38-
.checked_sub(other.t)
39-
.expect("second instant is later than self");
40-
Duration::new(diff / NSEC_PER_SEC, (diff % NSEC_PER_SEC) as u32)
36+
pub fn checked_sub_instant(&self, other: &Instant) -> Option<Duration> {
37+
let diff = self.t.checked_sub(other.t)?;
38+
Some(Duration::new(diff / NSEC_PER_SEC, (diff % NSEC_PER_SEC) as u32))
4139
}
4240

4341
pub fn checked_add_duration(&self, other: &Duration) -> Option<Instant> {

src/libstd/sys/redox/time.rs

+2-4
Original file line numberDiff line numberDiff line change
@@ -137,10 +137,8 @@ impl Instant {
137137
false
138138
}
139139

140-
pub fn sub_instant(&self, other: &Instant) -> Duration {
141-
self.t.sub_timespec(&other.t).unwrap_or_else(|_| {
142-
panic!("specified instant was later than self")
143-
})
140+
pub fn checked_sub_instant(&self, other: &Instant) -> Option<Duration> {
141+
self.t.sub_timespec(&other.t).ok()
144142
}
145143

146144
pub fn checked_add_duration(&self, other: &Duration) -> Option<Instant> {

src/libstd/sys/sgx/time.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -14,8 +14,8 @@ impl Instant {
1414
Instant(usercalls::insecure_time())
1515
}
1616

17-
pub fn sub_instant(&self, other: &Instant) -> Duration {
18-
self.0 - other.0
17+
pub fn checked_sub_instant(&self, other: &Instant) -> Option<Duration> {
18+
self.0.checked_sub(other.0)
1919
}
2020

2121
pub fn checked_add_duration(&self, other: &Duration) -> Option<Instant> {

src/libstd/sys/unix/time.rs

+5-8
Original file line numberDiff line numberDiff line change
@@ -149,12 +149,11 @@ mod inner {
149149
true
150150
}
151151

152-
pub fn sub_instant(&self, other: &Instant) -> Duration {
152+
pub fn checked_sub_instant(&self, other: &Instant) -> Option<Duration> {
153+
let diff = self.t.checked_sub(other.t)?;
153154
let info = info();
154-
let diff = self.t.checked_sub(other.t)
155-
.expect("second instant is later than self");
156155
let nanos = mul_div_u64(diff, info.numer as u64, info.denom as u64);
157-
Duration::new(nanos / NSEC_PER_SEC, (nanos % NSEC_PER_SEC) as u32)
156+
Some(Duration::new(nanos / NSEC_PER_SEC, (nanos % NSEC_PER_SEC) as u32))
158157
}
159158

160159
pub fn checked_add_duration(&self, other: &Duration) -> Option<Instant> {
@@ -285,10 +284,8 @@ mod inner {
285284
false // last clause, used so `||` is always trailing above
286285
}
287286

288-
pub fn sub_instant(&self, other: &Instant) -> Duration {
289-
self.t.sub_timespec(&other.t).unwrap_or_else(|_| {
290-
panic!("specified instant was later than self")
291-
})
287+
pub fn checked_sub_instant(&self, other: &Instant) -> Option<Duration> {
288+
self.t.sub_timespec(&other.t).ok()
292289
}
293290

294291
pub fn checked_add_duration(&self, other: &Duration) -> Option<Instant> {

src/libstd/sys/wasm/time.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -22,8 +22,8 @@ impl Instant {
2222
false
2323
}
2424

25-
pub fn sub_instant(&self, other: &Instant) -> Duration {
26-
self.0 - other.0
25+
pub fn checked_sub_instant(&self, other: &Instant) -> Option<Duration> {
26+
self.0.checked_sub(other.0)
2727
}
2828

2929
pub fn checked_add_duration(&self, other: &Duration) -> Option<Instant> {

src/libstd/sys/windows/time.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -49,17 +49,17 @@ impl Instant {
4949
Instant { t: Duration::from_secs(0) }
5050
}
5151

52-
pub fn sub_instant(&self, other: &Instant) -> Duration {
52+
pub fn checked_sub_instant(&self, other: &Instant) -> Option<Duration> {
5353
// On windows there's a threshold below which we consider two timestamps
5454
// equivalent due to measurement error. For more details + doc link,
5555
// check the docs on epsilon.
5656
let epsilon =
5757
perf_counter::PerformanceCounterInstant::epsilon();
5858
if other.t > self.t && other.t - self.t <= epsilon {
59-
return Duration::new(0, 0)
59+
Some(Duration::new(0, 0))
60+
} else {
61+
self.t.checked_sub(other.t)
6062
}
61-
self.t.checked_sub(other.t)
62-
.expect("specified instant was later than self")
6363
}
6464

6565
pub fn checked_add_duration(&self, other: &Duration) -> Option<Instant> {

src/libstd/time.rs

+11-12
Original file line numberDiff line numberDiff line change
@@ -212,7 +212,7 @@ impl Instant {
212212
/// ```
213213
#[stable(feature = "time2", since = "1.8.0")]
214214
pub fn duration_since(&self, earlier: Instant) -> Duration {
215-
self.0.sub_instant(&earlier.0)
215+
self.0.checked_sub_instant(&earlier.0).expect("supplied instant is later than self")
216216
}
217217

218218
/// Returns the amount of time elapsed from another instant to this one,
@@ -233,11 +233,7 @@ impl Instant {
233233
/// ```
234234
#[unstable(feature = "checked_duration_since", issue = "58402")]
235235
pub fn checked_duration_since(&self, earlier: Instant) -> Option<Duration> {
236-
if self >= &earlier {
237-
Some(self.0.sub_instant(&earlier.0))
238-
} else {
239-
None
240-
}
236+
self.0.checked_sub_instant(&earlier.0)
241237
}
242238

243239
/// Returns the amount of time elapsed from another instant to this one,
@@ -664,20 +660,23 @@ mod tests {
664660

665661
#[test]
666662
#[should_panic]
667-
fn instant_duration_panic() {
663+
fn instant_duration_since_panic() {
668664
let a = Instant::now();
669665
(a - Duration::new(1, 0)).duration_since(a);
670666
}
671667

672668
#[test]
673-
fn checked_instant_duration_nopanic() {
674-
let a = Instant::now();
675-
let ret = (a - Duration::new(1, 0)).checked_duration_since(a);
676-
assert_eq!(ret, None);
669+
fn instant_checked_duration_since_nopanic() {
670+
let now = Instant::now();
671+
let earlier = now - Duration::new(1, 0);
672+
let later = now + Duration::new(1, 0);
673+
assert_eq!(earlier.checked_duration_since(now), None);
674+
assert_eq!(later.checked_duration_since(now), Some(Duration::new(1, 0)));
675+
assert_eq!(now.checked_duration_since(now), Some(Duration::new(0, 0)));
677676
}
678677

679678
#[test]
680-
fn saturating_instant_duration_nopanic() {
679+
fn instant_saturating_duration_since_nopanic() {
681680
let a = Instant::now();
682681
let ret = (a - Duration::new(1, 0)).saturating_duration_since(a);
683682
assert_eq!(ret, Duration::new(0,0));

0 commit comments

Comments
 (0)