diff options
| author | Matthias Krüger <matthias.krueger@famsik.de> | 2022-02-20 00:37:30 +0100 |
|---|---|---|
| committer | GitHub <noreply@github.com> | 2022-02-20 00:37:30 +0100 |
| commit | 7cd857dc344e265aa0dd763355bd59e9609d33a1 (patch) | |
| tree | 2e3cc82f2fa47b88d9aa3b4e1ae0f9c019214814 | |
| parent | 39a50d829027c6377f288b9e96ff7382d98e5751 (diff) | |
| parent | 7c3ebec0caf23a11773c8291005649dd488ca2ee (diff) | |
| download | rust-7cd857dc344e265aa0dd763355bd59e9609d33a1.tar.gz rust-7cd857dc344e265aa0dd763355bd59e9609d33a1.zip | |
Rollup merge of #94112 - digama0:patch-3, r=scottmcm
Optimize char_try_from_u32 The optimization was proposed by ```````@falk-hueffner``````` in https://rust-lang.zulipchat.com/#narrow/stream/219381-t-libs/topic/Micro-optimizing.20char.3A.3Afrom_u32/near/272146171, and I simplified it a bit and added an explanation of why the optimization is correct. The generated code is 2 instructions shorter and uses 2 registers instead of 4 on x86.
| -rw-r--r-- | library/core/src/char/convert.rs | 17 |
1 files changed, 14 insertions, 3 deletions
diff --git a/library/core/src/char/convert.rs b/library/core/src/char/convert.rs index 1774ddd7cbb..139841368d6 100644 --- a/library/core/src/char/convert.rs +++ b/library/core/src/char/convert.rs @@ -6,8 +6,6 @@ use crate::fmt; use crate::mem::transmute; use crate::str::FromStr; -use super::MAX; - /// Converts a `u32` to a `char`. /// /// Note that all [`char`]s are valid [`u32`]s, and can be cast to one with @@ -271,7 +269,20 @@ impl FromStr for char { #[inline] const fn char_try_from_u32(i: u32) -> Result<char, CharTryFromError> { - if (i > MAX as u32) || (i >= 0xD800 && i <= 0xDFFF) { + // This is an optimized version of the check + // (i > MAX as u32) || (i >= 0xD800 && i <= 0xDFFF), + // which can also be written as + // i >= 0x110000 || (i >= 0xD800 && i < 0xE000). + // + // The XOR with 0xD800 permutes the ranges such that 0xD800..0xE000 is + // mapped to 0x0000..0x0800, while keeping all the high bits outside 0xFFFF the same. + // In particular, numbers >= 0x110000 stay in this range. + // + // Subtracting 0x800 causes 0x0000..0x0800 to wrap, meaning that a single + // unsigned comparison against 0x110000 - 0x800 will detect both the wrapped + // surrogate range as well as the numbers originally larger than 0x110000. + // + if (i ^ 0xD800).wrapping_sub(0x800) >= 0x110000 - 0x800 { Err(CharTryFromError(())) } else { // SAFETY: checked that it's a legal unicode value |
