-
Notifications
You must be signed in to change notification settings - Fork 4.8k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Removed JIT (uint) cast workaround to elide bound-checks in CoreLib #67448
Changes from 12 commits
cf4ccd4
af9b2d6
19b8e00
33cd7a1
20842e6
da31c0b
7e816a5
d9f4e01
034dd89
76d6d2a
186fb58
6afa6c2
26f459f
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,6 +1,8 @@ | ||
// Licensed to the .NET Foundation under one or more agreements. | ||
// The .NET Foundation licenses this file to you under the MIT license. | ||
|
||
using System.Diagnostics; | ||
|
||
namespace System.Collections.Generic | ||
{ | ||
internal ref struct BitHelper | ||
|
@@ -19,19 +21,29 @@ internal BitHelper(Span<int> span, bool clear) | |
|
||
internal void MarkBit(int bitPosition) | ||
{ | ||
int bitArrayIndex = bitPosition / IntSize; | ||
if ((uint)bitArrayIndex < (uint)_span.Length) | ||
Debug.Assert(bitPosition >= 0); | ||
|
||
uint bitArrayIndex = (uint)bitPosition / IntSize; | ||
|
||
// Workaround for https://github.com/dotnet/runtime/issues/72004 | ||
Span<int> span = _span; | ||
if (bitArrayIndex < (uint)span.Length) | ||
{ | ||
_span[bitArrayIndex] |= (1 << (bitPosition % IntSize)); | ||
span[(int)bitArrayIndex] |= (1 << (int)((uint)bitPosition % IntSize)); | ||
} | ||
} | ||
|
||
internal bool IsMarked(int bitPosition) | ||
{ | ||
int bitArrayIndex = bitPosition / IntSize; | ||
Debug.Assert(bitPosition >= 0); | ||
|
||
uint bitArrayIndex = (uint)bitPosition / IntSize; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Same for this one, its doing the same thing but via "more code" now. I'd also expect the JIT was already hoisting the span since its a field of a struct. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It is not the same thing. Unsigned division is a simple shift. Signed division is more complicated. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Unfortunately that's not the case*. Codegen was validated for this change. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The operation is "the same" even if the codegen is different. If we're doing this specifically to take advantage of the codegen, there should be a comment on it. More generally, we also need some assert that the input is definitely positive -or- should be taking the input as If we're going to be making the method less readable, we should include a comment explaining why these tricks are being used (just as all the current cases have a small |
||
|
||
// Workaround for https://github.com/dotnet/runtime/issues/72004 | ||
Span<int> span = _span; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This one can use a tracking issue. The JIT should be smart enough to make it unnecessary to cache the Span here. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Filed #72004 for this. |
||
return | ||
(uint)bitArrayIndex < (uint)_span.Length && | ||
(_span[bitArrayIndex] & (1 << (bitPosition % IntSize))) != 0; | ||
bitArrayIndex < (uint)span.Length && | ||
(span[(int)bitArrayIndex] & (1 << ((int)((uint)bitPosition % IntSize)))) != 0; | ||
} | ||
|
||
/// <summary>How many ints must be allocated to represent n bits. Returns (n+31)/32, but avoids overflow.</summary> | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -591,7 +591,7 @@ public static bool IsControl(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. These ones seem to be touching code "stylistically", we probably should exclude it. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. W/o the superfluos paranthesis it's easier to read, thus I changed it. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't disagree, but its also a "stylistic" change and so makes it harder to review the thing we actually care about. The typical PR policy is to not include such things and leave them to a separate PR, especially when its the only or primary change to a given file.
We should have issues tracking these. We really shouldn't be getting codegen differences between |
||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -606,7 +606,7 @@ public static bool IsDigit(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -626,7 +626,7 @@ public static bool IsLetter(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -647,7 +647,7 @@ public static bool IsLetterOrDigit(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -667,8 +667,7 @@ public static bool IsLower(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
|
||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -710,7 +709,7 @@ public static bool IsNumber(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -743,7 +742,7 @@ public static bool IsPunctuation(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -788,7 +787,7 @@ public static bool IsSeparator(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -813,7 +812,7 @@ public static bool IsSurrogate(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -845,7 +844,7 @@ public static bool IsSymbol(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -865,7 +864,7 @@ public static bool IsUpper(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -885,7 +884,7 @@ public static bool IsWhiteSpace(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -911,7 +910,7 @@ public static UnicodeCategory GetUnicodeCategory(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -935,7 +934,7 @@ public static double GetNumericValue(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -957,7 +956,7 @@ public static bool IsHighSurrogate(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -979,7 +978,7 @@ public static bool IsLowSurrogate(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
@@ -996,7 +995,7 @@ public static bool IsSurrogatePair(string s, int index) | |
{ | ||
ThrowHelper.ThrowArgumentNullException(ExceptionArgument.s); | ||
} | ||
if (((uint)index) >= ((uint)s.Length)) | ||
if ((uint)index >= (uint)s.Length) | ||
{ | ||
ThrowHelper.ThrowArgumentOutOfRangeException(ExceptionArgument.index); | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can we add a
Debug.Assert(bitPosition >= 0)
? Same for the other two methods here.The previous logic, would've done say
-1 / 32 == 0
and now will douint.MaxValue / 32 = 134217727
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It might even be better to just keep this one "as is" since its really doing the same thing, just with more casts now (that is comparing
uint
touint
)