Skip to content

Commit ffdf18d

Browse files
committed
Auto merge of #88788 - falk-hueffner:speedup-int-log10-branchless, r=joshtriplett
Speedup int log10 branchless This is achieved with a branchless bit-twiddling implementation of the case x < 100_000, and using this as building block. Benchmark on an Intel i7-8700K (Coffee Lake): ``` name old ns/iter new ns/iter diff ns/iter diff % speedup num::int_log::u8_log10_predictable 165 169 4 2.42% x 0.98 num::int_log::u8_log10_random 438 423 -15 -3.42% x 1.04 num::int_log::u8_log10_random_small 438 423 -15 -3.42% x 1.04 num::int_log::u16_log10_predictable 633 417 -216 -34.12% x 1.52 num::int_log::u16_log10_random 908 471 -437 -48.13% x 1.93 num::int_log::u16_log10_random_small 945 471 -474 -50.16% x 2.01 num::int_log::u32_log10_predictable 1,496 1,340 -156 -10.43% x 1.12 num::int_log::u32_log10_random 1,076 873 -203 -18.87% x 1.23 num::int_log::u32_log10_random_small 1,145 874 -271 -23.67% x 1.31 num::int_log::u64_log10_predictable 4,005 3,171 -834 -20.82% x 1.26 num::int_log::u64_log10_random 1,247 1,021 -226 -18.12% x 1.22 num::int_log::u64_log10_random_small 1,265 921 -344 -27.19% x 1.37 num::int_log::u128_log10_predictable 39,667 39,579 -88 -0.22% x 1.00 num::int_log::u128_log10_random 6,456 6,696 240 3.72% x 0.96 num::int_log::u128_log10_random_small 4,108 3,903 -205 -4.99% x 1.05 ``` Benchmark on an M1 Mac Mini: ``` name old ns/iter new ns/iter diff ns/iter diff % speedup num::int_log::u8_log10_predictable 143 130 -13 -9.09% x 1.10 num::int_log::u8_log10_random 375 325 -50 -13.33% x 1.15 num::int_log::u8_log10_random_small 376 325 -51 -13.56% x 1.16 num::int_log::u16_log10_predictable 500 322 -178 -35.60% x 1.55 num::int_log::u16_log10_random 794 405 -389 -48.99% x 1.96 num::int_log::u16_log10_random_small 1,035 405 -630 -60.87% x 2.56 num::int_log::u32_log10_predictable 1,144 894 -250 -21.85% x 1.28 num::int_log::u32_log10_random 832 786 -46 -5.53% x 1.06 num::int_log::u32_log10_random_small 832 787 -45 -5.41% x 1.06 num::int_log::u64_log10_predictable 2,681 2,057 -624 -23.27% x 1.30 num::int_log::u64_log10_random 1,015 806 -209 -20.59% x 1.26 num::int_log::u64_log10_random_small 1,004 795 -209 -20.82% x 1.26 num::int_log::u128_log10_predictable 56,825 56,526 -299 -0.53% x 1.01 num::int_log::u128_log10_random 9,056 8,861 -195 -2.15% x 1.02 num::int_log::u128_log10_random_small 1,528 1,527 -1 -0.07% x 1.00 ``` The 128 bit case remains ridiculously slow because llvm fails to optimize division by a constant 128-bit value to multiplications. This could be worked around but it seems preferable to fix this in llvm. From u32 up, table lookup (like suggested [here](#70887 (comment))) is still faster, but requires a hardware `leading_zeros` to be viable, and might clog up the cache.
2 parents 97e3b30 + 57c6235 commit ffdf18d

File tree

5 files changed

+114
-56
lines changed

5 files changed

+114
-56
lines changed

library/core/benches/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
// wasm32 does not support benches (no time).
22
#![cfg(not(target_arch = "wasm32"))]
33
#![feature(flt2dec)]
4+
#![feature(int_log)]
45
#![feature(test)]
56

67
extern crate test;
+58
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
use rand::Rng;
2+
use test::{black_box, Bencher};
3+
4+
macro_rules! int_log_bench {
5+
($t:ty, $predictable:ident, $random:ident, $random_small:ident) => {
6+
#[bench]
7+
fn $predictable(bench: &mut Bencher) {
8+
bench.iter(|| {
9+
for n in 0..(<$t>::BITS / 8) {
10+
for i in 1..=(100 as $t) {
11+
let x = black_box(i << (n * 8));
12+
black_box(x.log10());
13+
}
14+
}
15+
});
16+
}
17+
18+
#[bench]
19+
fn $random(bench: &mut Bencher) {
20+
let mut rng = rand::thread_rng();
21+
/* Exponentially distributed random numbers from the whole range of the type. */
22+
let numbers: Vec<$t> = (0..256)
23+
.map(|_| {
24+
let x = rng.gen::<$t>() >> rng.gen_range(0, <$t>::BITS);
25+
if x != 0 { x } else { 1 }
26+
})
27+
.collect();
28+
bench.iter(|| {
29+
for x in &numbers {
30+
black_box(black_box(x).log10());
31+
}
32+
});
33+
}
34+
35+
#[bench]
36+
fn $random_small(bench: &mut Bencher) {
37+
let mut rng = rand::thread_rng();
38+
/* Exponentially distributed random numbers from the range 0..256. */
39+
let numbers: Vec<$t> = (0..256)
40+
.map(|_| {
41+
let x = (rng.gen::<u8>() >> rng.gen_range(0, u8::BITS)) as $t;
42+
if x != 0 { x } else { 1 }
43+
})
44+
.collect();
45+
bench.iter(|| {
46+
for x in &numbers {
47+
black_box(black_box(x).log10());
48+
}
49+
});
50+
}
51+
};
52+
}
53+
54+
int_log_bench! {u8, u8_log10_predictable, u8_log10_random, u8_log10_random_small}
55+
int_log_bench! {u16, u16_log10_predictable, u16_log10_random, u16_log10_random_small}
56+
int_log_bench! {u32, u32_log10_predictable, u32_log10_random, u32_log10_random_small}
57+
int_log_bench! {u64, u64_log10_predictable, u64_log10_random, u64_log10_random_small}
58+
int_log_bench! {u128, u128_log10_predictable, u128_log10_random, u128_log10_random_small}

library/core/benches/num/mod.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
mod dec2flt;
22
mod flt2dec;
3+
mod int_log;
34

45
use std::str::FromStr;
56
use test::Bencher;

library/core/src/num/int_log10.rs

+51-56
Original file line numberDiff line numberDiff line change
@@ -1,76 +1,71 @@
11
mod unchecked {
22
// 0 < val <= u8::MAX
33
pub const fn u8(val: u8) -> u32 {
4-
if val >= 100 {
5-
2
6-
} else if val >= 10 {
7-
1
8-
} else {
9-
0
10-
}
4+
let val = val as u32;
5+
6+
// For better performance, avoid branches by assembling the solution
7+
// in the bits above the low 8 bits.
8+
9+
// Adding c1 to val gives 10 in the top bits for val < 10, 11 for val >= 10
10+
const C1: u32 = 0b11_00000000 - 10; // 758
11+
// Adding c2 to val gives 01 in the top bits for val < 100, 10 for val >= 100
12+
const C2: u32 = 0b10_00000000 - 100; // 412
13+
14+
// Value of top bits:
15+
// +c1 +c2 1&2
16+
// 0..=9 10 01 00 = 0
17+
// 10..=99 11 01 01 = 1
18+
// 100..=255 11 10 10 = 2
19+
((val + C1) & (val + C2)) >> 8
1120
}
1221

13-
// 0 < val <= u16::MAX
14-
pub const fn u16(val: u16) -> u32 {
15-
if val >= 10_000 {
16-
4
17-
} else if val >= 1000 {
18-
3
19-
} else if val >= 100 {
20-
2
21-
} else if val >= 10 {
22-
1
23-
} else {
24-
0
25-
}
22+
// 0 < val < 100_000
23+
const fn less_than_5(val: u32) -> u32 {
24+
// Similar to u8, when adding one of these constants to val,
25+
// we get two possible bit patterns above the low 17 bits,
26+
// depending on whether val is below or above the threshold.
27+
const C1: u32 = 0b011_00000000000000000 - 10; // 393206
28+
const C2: u32 = 0b100_00000000000000000 - 100; // 524188
29+
const C3: u32 = 0b111_00000000000000000 - 1000; // 916504
30+
const C4: u32 = 0b100_00000000000000000 - 10000; // 514288
31+
32+
// Value of top bits:
33+
// +c1 +c2 1&2 +c3 +c4 3&4 ^
34+
// 0..=9 010 011 010 110 011 010 000 = 0
35+
// 10..=99 011 011 011 110 011 010 001 = 1
36+
// 100..=999 011 100 000 110 011 010 010 = 2
37+
// 1000..=9999 011 100 000 111 011 011 011 = 3
38+
// 10000..=99999 011 100 000 111 100 100 100 = 4
39+
(((val + C1) & (val + C2)) ^ ((val + C3) & (val + C4))) >> 17
2640
}
2741

28-
// 0 < val < 100_000_000
29-
const fn less_than_8(mut val: u32) -> u32 {
30-
let mut log = 0;
31-
if val >= 10_000 {
32-
val /= 10_000;
33-
log += 4;
34-
}
35-
log + if val >= 1000 {
36-
3
37-
} else if val >= 100 {
38-
2
39-
} else if val >= 10 {
40-
1
41-
} else {
42-
0
43-
}
42+
// 0 < val <= u16::MAX
43+
pub const fn u16(val: u16) -> u32 {
44+
less_than_5(val as u32)
4445
}
4546

4647
// 0 < val <= u32::MAX
4748
pub const fn u32(mut val: u32) -> u32 {
4849
let mut log = 0;
49-
if val >= 100_000_000 {
50-
val /= 100_000_000;
51-
log += 8;
52-
}
53-
log + less_than_8(val)
54-
}
55-
56-
// 0 < val < 10_000_000_000_000_000
57-
const fn less_than_16(mut val: u64) -> u32 {
58-
let mut log = 0;
59-
if val >= 100_000_000 {
60-
val /= 100_000_000;
61-
log += 8;
50+
if val >= 100_000 {
51+
val /= 100_000;
52+
log += 5;
6253
}
63-
log + less_than_8(val as u32)
54+
log + less_than_5(val)
6455
}
6556

6657
// 0 < val <= u64::MAX
6758
pub const fn u64(mut val: u64) -> u32 {
6859
let mut log = 0;
69-
if val >= 10_000_000_000_000_000 {
70-
val /= 10_000_000_000_000_000;
71-
log += 16;
60+
if val >= 10_000_000_000 {
61+
val /= 10_000_000_000;
62+
log += 10;
63+
}
64+
if val >= 100_000 {
65+
val /= 100_000;
66+
log += 5;
7267
}
73-
log + less_than_16(val)
68+
log + less_than_5(val as u32)
7469
}
7570

7671
// 0 < val <= u128::MAX
@@ -79,13 +74,13 @@ mod unchecked {
7974
if val >= 100_000_000_000_000_000_000_000_000_000_000 {
8075
val /= 100_000_000_000_000_000_000_000_000_000_000;
8176
log += 32;
82-
return log + less_than_8(val as u32);
77+
return log + u32(val as u32);
8378
}
8479
if val >= 10_000_000_000_000_000 {
8580
val /= 10_000_000_000_000_000;
8681
log += 16;
8782
}
88-
log + less_than_16(val as u64)
83+
log + u64(val as u64)
8984
}
9085

9186
// 0 < val <= i8::MAX

library/core/tests/num/int_log.rs

+3
Original file line numberDiff line numberDiff line change
@@ -96,6 +96,9 @@ fn checked_log10() {
9696
for i in 1..=u16::MAX {
9797
assert_eq!(i.checked_log10(), Some((i as f32).log10() as u32));
9898
}
99+
for i in 1..=100_000u32 {
100+
assert_eq!(i.checked_log10(), Some((i as f32).log10() as u32));
101+
}
99102
}
100103

101104
macro_rules! log10_loop {

0 commit comments

Comments
 (0)