Skip to content

Commit 6e33ec9

Browse files
authored
Removed JIT (uint) cast workaround to elide bound-checks in CoreLib (#67448)
* Removed (uint) cast workaround in CoreLib, annotated missing places with TODOs * Fixed the bug I introduced in TimeSpanParse.TimeSpanTokenizer * Use uint-cast on both sided for clarity Cf. #67448 (comment) * Use uint-division in BitHelper as the JIT can produce more efficient code Cf. #67448 (comment) * Don't have struct local in ValueStringBuilder due to hitting JIT optimization limits Cf. #67448 (comment) * BitHelper with uint division and without Math.DivRem for even better codegen AggressiveInlining isn't needed anymore, also the JIT recognizes the _span field, so no local Span is needed here. * BitHelper needs local Span due to regression in jit-diff otherwise * Added Debug.Asserts to BitHelper's method that take bitPosition * BitHelper: comment for workaround to tracking issue
1 parent 608e3fd commit 6e33ec9

File tree

15 files changed

+89
-77
lines changed

15 files changed

+89
-77
lines changed

src/libraries/Common/src/System/Collections/Generic/BitHelper.cs

+18-6
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
// Licensed to the .NET Foundation under one or more agreements.
22
// The .NET Foundation licenses this file to you under the MIT license.
33

4+
using System.Diagnostics;
5+
46
namespace System.Collections.Generic
57
{
68
internal ref struct BitHelper
@@ -19,19 +21,29 @@ internal BitHelper(Span<int> span, bool clear)
1921

2022
internal void MarkBit(int bitPosition)
2123
{
22-
int bitArrayIndex = bitPosition / IntSize;
23-
if ((uint)bitArrayIndex < (uint)_span.Length)
24+
Debug.Assert(bitPosition >= 0);
25+
26+
uint bitArrayIndex = (uint)bitPosition / IntSize;
27+
28+
// Workaround for https://github.com/dotnet/runtime/issues/72004
29+
Span<int> span = _span;
30+
if (bitArrayIndex < (uint)span.Length)
2431
{
25-
_span[bitArrayIndex] |= (1 << (bitPosition % IntSize));
32+
span[(int)bitArrayIndex] |= (1 << (int)((uint)bitPosition % IntSize));
2633
}
2734
}
2835

2936
internal bool IsMarked(int bitPosition)
3037
{
31-
int bitArrayIndex = bitPosition / IntSize;
38+
Debug.Assert(bitPosition >= 0);
39+
40+
uint bitArrayIndex = (uint)bitPosition / IntSize;
41+
42+
// Workaround for https://github.com/dotnet/runtime/issues/72004
43+
Span<int> span = _span;
3244
return
33-
(uint)bitArrayIndex < (uint)_span.Length &&
34-
(_span[bitArrayIndex] & (1 << (bitPosition % IntSize))) != 0;
45+
bitArrayIndex < (uint)span.Length &&
46+
(span[(int)bitArrayIndex] & (1 << ((int)((uint)bitPosition % IntSize)))) != 0;
3547
}
3648

3749
/// <summary>How many ints must be allocated to represent n bits. Returns (n+31)/32, but avoids overflow.</summary>

src/libraries/System.Private.CoreLib/src/System/Boolean.cs

+2-2
Original file line numberDiff line numberDiff line change
@@ -99,7 +99,7 @@ public bool TryFormat(Span<char> destination, out int charsWritten)
9999
{
100100
if (m_value)
101101
{
102-
if ((uint)destination.Length > 3) // uint cast, per https://github.com/dotnet/runtime/issues/10596
102+
if (destination.Length > 3)
103103
{
104104
ulong true_val = BitConverter.IsLittleEndian ? 0x65007500720054ul : 0x54007200750065ul; // "True"
105105
MemoryMarshal.Write<ulong>(MemoryMarshal.AsBytes(destination), ref true_val);
@@ -109,7 +109,7 @@ public bool TryFormat(Span<char> destination, out int charsWritten)
109109
}
110110
else
111111
{
112-
if ((uint)destination.Length > 4)
112+
if (destination.Length > 4)
113113
{
114114
ulong fals_val = BitConverter.IsLittleEndian ? 0x73006C00610046ul : 0x460061006C0073ul; // "Fals"
115115
MemoryMarshal.Write<ulong>(MemoryMarshal.AsBytes(destination), ref fals_val);

src/libraries/System.Private.CoreLib/src/System/Buffers/StandardFormat.cs

+1-2
Original file line numberDiff line numberDiff line change
@@ -164,8 +164,7 @@ internal int Format(Span<char> destination)
164164
int count = 0;
165165
char symbol = Symbol;
166166

167-
if (symbol != default &&
168-
(uint)destination.Length == FormatStringLength) // to eliminate bounds checks
167+
if (symbol != default && destination.Length == FormatStringLength)
169168
{
170169
destination[0] = symbol;
171170
count = 1;

src/libraries/System.Private.CoreLib/src/System/Buffers/Text/Utf8Formatter/Utf8Formatter.Boolean.cs

+2-2
Original file line numberDiff line numberDiff line change
@@ -65,7 +65,7 @@ public static bool TryFormat(bool value, Span<byte> destination, out int bytesWr
6565
{
6666
// This check can't be performed earlier because we need to throw if an invalid symbol is
6767
// provided, even if the buffer is too small.
68-
if ((uint)4 >= (uint)destination.Length)
68+
if (destination.Length <= 4)
6969
{
7070
goto BufferTooSmall;
7171
}
@@ -75,7 +75,7 @@ public static bool TryFormat(bool value, Span<byte> destination, out int bytesWr
7575
}
7676
else if (symbol == 'l')
7777
{
78-
if ((uint)4 >= (uint)destination.Length)
78+
if (destination.Length <= 4)
7979
{
8080
goto BufferTooSmall;
8181
}

src/libraries/System.Private.CoreLib/src/System/Buffers/Text/Utf8Formatter/Utf8Formatter.Date.L.cs

+1-3
Original file line numberDiff line numberDiff line change
@@ -13,9 +13,7 @@ public static partial class Utf8Formatter
1313
//
1414
private static bool TryFormatDateTimeL(DateTime value, Span<byte> destination, out int bytesWritten)
1515
{
16-
// Writing the check in this fashion elides all bounds checks on 'buffer'
17-
// for the remainder of the method.
18-
if ((uint)28 >= (uint)destination.Length)
16+
if (destination.Length <= 28)
1917
{
2018
bytesWritten = 0;
2119
return false;

src/libraries/System.Private.CoreLib/src/System/Buffers/Text/Utf8Formatter/Utf8Formatter.Date.R.cs

+1-3
Original file line numberDiff line numberDiff line change
@@ -13,9 +13,7 @@ public static partial class Utf8Formatter
1313
//
1414
private static bool TryFormatDateTimeR(DateTime value, Span<byte> destination, out int bytesWritten)
1515
{
16-
// Writing the check in this fashion elides all bounds checks on 'buffer'
17-
// for the remainder of the method.
18-
if ((uint)28 >= (uint)destination.Length)
16+
if (destination.Length <= 28)
1917
{
2018
bytesWritten = 0;
2119
return false;

src/libraries/System.Private.CoreLib/src/System/Char.cs

+17-18
Original file line numberDiff line numberDiff line change
@@ -591,7 +591,7 @@ public static bool IsControl(string s, int index)
591591
{
592592
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
593593
}
594-
if (((uint)index) >= ((uint)s.Length))
594+
if ((uint)index >= (uint)s.Length)
595595
{
596596
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
597597
}
@@ -606,7 +606,7 @@ public static bool IsDigit(string s, int index)
606606
{
607607
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
608608
}
609-
if (((uint)index) >= ((uint)s.Length))
609+
if ((uint)index >= (uint)s.Length)
610610
{
611611
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
612612
}
@@ -626,7 +626,7 @@ public static bool IsLetter(string s, int index)
626626
{
627627
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
628628
}
629-
if (((uint)index) >= ((uint)s.Length))
629+
if ((uint)index >= (uint)s.Length)
630630
{
631631
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
632632
}
@@ -647,7 +647,7 @@ public static bool IsLetterOrDigit(string s, int index)
647647
{
648648
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
649649
}
650-
if (((uint)index) >= ((uint)s.Length))
650+
if ((uint)index >= (uint)s.Length)
651651
{
652652
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
653653
}
@@ -667,8 +667,7 @@ public static bool IsLower(string s, int index)
667667
{
668668
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
669669
}
670-
671-
if (((uint)index) >= ((uint)s.Length))
670+
if ((uint)index >= (uint)s.Length)
672671
{
673672
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
674673
}
@@ -710,7 +709,7 @@ public static bool IsNumber(string s, int index)
710709
{
711710
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
712711
}
713-
if (((uint)index) >= ((uint)s.Length))
712+
if ((uint)index >= (uint)s.Length)
714713
{
715714
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
716715
}
@@ -743,7 +742,7 @@ public static bool IsPunctuation(string s, int index)
743742
{
744743
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
745744
}
746-
if (((uint)index) >= ((uint)s.Length))
745+
if ((uint)index >= (uint)s.Length)
747746
{
748747
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
749748
}
@@ -788,7 +787,7 @@ public static bool IsSeparator(string s, int index)
788787
{
789788
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
790789
}
791-
if (((uint)index) >= ((uint)s.Length))
790+
if ((uint)index >= (uint)s.Length)
792791
{
793792
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
794793
}
@@ -813,7 +812,7 @@ public static bool IsSurrogate(string s, int index)
813812
{
814813
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
815814
}
816-
if (((uint)index) >= ((uint)s.Length))
815+
if ((uint)index >= (uint)s.Length)
817816
{
818817
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
819818
}
@@ -845,7 +844,7 @@ public static bool IsSymbol(string s, int index)
845844
{
846845
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
847846
}
848-
if (((uint)index) >= ((uint)s.Length))
847+
if ((uint)index >= (uint)s.Length)
849848
{
850849
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
851850
}
@@ -865,7 +864,7 @@ public static bool IsUpper(string s, int index)
865864
{
866865
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
867866
}
868-
if (((uint)index) >= ((uint)s.Length))
867+
if ((uint)index >= (uint)s.Length)
869868
{
870869
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
871870
}
@@ -885,7 +884,7 @@ public static bool IsWhiteSpace(string s, int index)
885884
{
886885
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
887886
}
888-
if (((uint)index) >= ((uint)s.Length))
887+
if ((uint)index >= (uint)s.Length)
889888
{
890889
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
891890
}
@@ -911,7 +910,7 @@ public static UnicodeCategory GetUnicodeCategory(string s, int index)
911910
{
912911
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
913912
}
914-
if (((uint)index) >= ((uint)s.Length))
913+
if ((uint)index >= (uint)s.Length)
915914
{
916915
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
917916
}
@@ -935,7 +934,7 @@ public static double GetNumericValue(string s, int index)
935934
{
936935
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
937936
}
938-
if (((uint)index) >= ((uint)s.Length))
937+
if ((uint)index >= (uint)s.Length)
939938
{
940939
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
941940
}
@@ -957,7 +956,7 @@ public static bool IsHighSurrogate(string s, int index)
957956
{
958957
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
959958
}
960-
if (((uint)index) >= ((uint)s.Length))
959+
if ((uint)index >= (uint)s.Length)
961960
{
962961
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
963962
}
@@ -979,7 +978,7 @@ public static bool IsLowSurrogate(string s, int index)
979978
{
980979
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
981980
}
982-
if (((uint)index) >= ((uint)s.Length))
981+
if ((uint)index >= (uint)s.Length)
983982
{
984983
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
985984
}
@@ -996,7 +995,7 @@ public static bool IsSurrogatePair(string s, int index)
996995
{
997996
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s);
998997
}
999-
if (((uint)index) >= ((uint)s.Length))
998+
if ((uint)index >= (uint)s.Length)
1000999
{
10011000
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index);
10021001
}

src/libraries/System.Private.CoreLib/src/System/Collections/Generic/ValueListBuilder.cs

+5-2
Original file line numberDiff line numberDiff line change
@@ -44,9 +44,12 @@ public ref T this[int index]
4444
public void Append(T item)
4545
{
4646
int pos = _pos;
47-
if ((uint)pos < (uint)_span.Length)
47+
48+
// Workaround for https://github.com/dotnet/runtime/issues/72004
49+
Span<T> span = _span;
50+
if ((uint)pos < (uint)span.Length)
4851
{
49-
_span[pos] = item;
52+
span[pos] = item;
5053
_pos = pos + 1;
5154
}
5255
else

src/libraries/System.Private.CoreLib/src/System/Decimal.cs

+2-2
Original file line numberDiff line numberDiff line change
@@ -602,7 +602,7 @@ public static int[] GetBits(decimal d)
602602
/// <exception cref="ArgumentException">The destination span was not long enough to store the binary representation.</exception>
603603
public static int GetBits(decimal d, Span<int> destination)
604604
{
605-
if ((uint)destination.Length <= 3)
605+
if (destination.Length <= 3)
606606
{
607607
ThrowHelper.ThrowArgumentException_DestinationTooShort();
608608
}
@@ -623,7 +623,7 @@ public static int GetBits(decimal d, Span<int> destination)
623623
/// <returns>true if the decimal's binary representation was written to the destination; false if the destination wasn't long enough.</returns>
624624
public static bool TryGetBits(decimal d, Span<int> destination, out int valuesWritten)
625625
{
626-
if ((uint)destination.Length <= 3)
626+
if (destination.Length <= 3)
627627
{
628628
valuesWritten = 0;
629629
return false;

src/libraries/System.Private.CoreLib/src/System/Globalization/DateTimeParse.cs

+5-5
Original file line numberDiff line numberDiff line change
@@ -4663,7 +4663,7 @@ private static bool ParseFormatR(ReadOnlySpan<char> source, ref ParsingInfo pars
46634663
// Tue, 03 Jan 2017 08:08:05 GMT
46644664

46654665
// The format is exactly 29 characters.
4666-
if ((uint)source.Length != 29)
4666+
if (source.Length != 29)
46674667
{
46684668
result.SetBadDateTimeFailure();
46694669
return false;
@@ -4860,7 +4860,7 @@ private static bool ParseFormatO(ReadOnlySpan<char> source, ref DateTimeResult r
48604860
// 2017-06-12T05:30:45.7680000-7:00 (special-case of one-digit offset hour)
48614861
// 2017-06-12T05:30:45.7680000-07:00
48624862

4863-
if ((uint)source.Length < 27 ||
4863+
if (source.Length < 27 ||
48644864
source[4] != '-' ||
48654865
source[7] != '-' ||
48664866
source[10] != 'T' ||
@@ -4981,7 +4981,7 @@ private static bool ParseFormatO(ReadOnlySpan<char> source, ref DateTimeResult r
49814981
return false;
49824982
}
49834983

4984-
if ((uint)source.Length > 27)
4984+
if (source.Length > 27)
49854985
{
49864986
char offsetChar = source[27];
49874987
switch (offsetChar)
@@ -4999,7 +4999,7 @@ private static bool ParseFormatO(ReadOnlySpan<char> source, ref DateTimeResult r
49994999
case '-':
50005000
int offsetHours, colonIndex;
50015001

5002-
if ((uint)source.Length == 33)
5002+
if (source.Length == 33)
50035003
{
50045004
uint oh1 = (uint)(source[28] - '0'), oh2 = (uint)(source[29] - '0');
50055005

@@ -5012,7 +5012,7 @@ private static bool ParseFormatO(ReadOnlySpan<char> source, ref DateTimeResult r
50125012
offsetHours = (int)(oh1 * 10 + oh2);
50135013
colonIndex = 30;
50145014
}
5015-
else if ((uint)source.Length == 32) // special-case allowed for compat: only one offset hour digit
5015+
else if (source.Length == 32) // special-case allowed for compat: only one offset hour digit
50165016
{
50175017
offsetHours = source[28] - '0';
50185018

0 commit comments

Comments
 (0)