Skip to content
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

Unnecessary sign-extension for (nuint)span.Length #84564

Closed
xtqqczze opened this issue Apr 10, 2023 · 3 comments · Fixed by #88136
Closed

Unnecessary sign-extension for (nuint)span.Length #84564

xtqqczze opened this issue Apr 10, 2023 · 3 comments · Fixed by #88136
Labels
area-CodeGen-coreclr CLR JIT compiler in src/coreclr/src/jit and related components such as SuperPMI help wanted [up-for-grabs] Good issue for external contributors
Milestone

Comments

@xtqqczze
Copy link
Contributor

xtqqczze commented Apr 10, 2023

@EgorBo In theory, the length in (nuint)span.Length should be recognized as never negative, and the sign-extension elided, but doesn't appear to be the case (at least in R2R), see https://csharp.godbolt.org/z/14oW68fqz.

Originally posted by @xtqqczze in #84524 (comment)

static nuint Span0(ReadOnlySpan<byte> span) => (nuint)span.Length;
static nuint Span1(ReadOnlySpan<byte> span) => (uint)span.Length;
// crossgen2 8.0.0-preview.4.23208.99+07f00dc2f624946c54dbf85fea79c600a690f634

C:Span0(System.ReadOnlySpan`1[ubyte]):ulong:
; Emitting BLENDED_CODE for X64 CPU with AVX - Windows
						;; size=0 bbWeight=1 PerfScore 0.00
       movsxd   rax, dword ptr [rcx+08H]
						;; size=4 bbWeight=1 PerfScore 4.00
       ret      
						;; size=1 bbWeight=1 PerfScore 1.00

C:Span1(System.ReadOnlySpan`1[ubyte]):ulong:
; Emitting BLENDED_CODE for X64 CPU with AVX - Windows
						;; size=0 bbWeight=1 PerfScore 0.00
       mov      eax, dword ptr [rcx+08H]
						;; size=3 bbWeight=1 PerfScore 2.00
       ret      
						;; size=1 bbWeight=1 PerfScore 1.00
@dotnet-issue-labeler dotnet-issue-labeler bot added the needs-area-label An area label is needed to ensure this gets routed to the appropriate area owners label Apr 10, 2023
@ghost ghost added the untriaged New issue has not been triaged by the area owner label Apr 10, 2023
@EgorBo EgorBo added area-CodeGen-coreclr CLR JIT compiler in src/coreclr/src/jit and related components such as SuperPMI and removed needs-area-label An area label is needed to ensure this gets routed to the appropriate area owners untriaged New issue has not been triaged by the area owner labels Apr 10, 2023
@EgorBo EgorBo added this to the Future milestone Apr 10, 2023
@ghost
Copy link

ghost commented Apr 10, 2023

Tagging subscribers to this area: @JulieLeeMSFT, @jakobbotsch, @kunalspathak
See info in area-owners.md if you want to be subscribed.

Issue Details

@EgorBo In theory, the length in (nuint)span.Length should be recognized as never negative, and the sign-extension elided, but doesn't appear to be the case (at least in R2R), see https://csharp.godbolt.org/z/14oW68fqz.

Originally posted by @xtqqczze in #84524 (comment)

static nuint Span0(ReadOnlySpan<byte> span) => (nuint)span.Length;
static nuint Span1(ReadOnlySpan<byte> span) => (uint)span.Length;
// crossgen2 8.0.0-preview.4.23208.99+07f00dc2f624946c54dbf85fea79c600a690f634

C:Span0(System.ReadOnlySpan`1[ubyte]):ulong:
; Emitting BLENDED_CODE for X64 CPU with AVX - Windows
						;; size=0 bbWeight=1 PerfScore 0.00
       movsxd   rax, dword ptr [rcx+08H]
						;; size=4 bbWeight=1 PerfScore 4.00
       ret      
						;; size=1 bbWeight=1 PerfScore 1.00

C:Span1(System.ReadOnlySpan`1[ubyte]):ulong:
; Emitting BLENDED_CODE for X64 CPU with AVX - Windows
						;; size=0 bbWeight=1 PerfScore 0.00
       mov      eax, dword ptr [rcx+08H]
						;; size=3 bbWeight=1 PerfScore 2.00
       ret      
						;; size=1 bbWeight=1 PerfScore 1.00
Author: xtqqczze
Assignees: -
Labels:

area-CodeGen-coreclr, untriaged

Milestone: -

@EgorBo EgorBo added the help wanted [up-for-grabs] Good issue for external contributors label Apr 10, 2023
@EgorBo
Copy link
Member

EgorBo commented Apr 10, 2023

Should be a simple fix since Span.Length can be recognized via IsNeverNegative. We just need to mark such GT_CAST as GTF_UNSIGNED if its target is IsNeverNegative and we're upcasting

@SingleAccretion
Copy link
Contributor

We already do:

runtime/src/coreclr/jit/morph.cpp

Lines 10190 to 10194 in 00dbf84

// Try and see if we can make this cast into a cheaper zero-extending version.
if (genActualTypeIsInt(src) && cast->TypeIs(TYP_LONG) && srcRange.IsNonNegative())
{
cast->SetUnsigned();
}

The snippets in the issue are cases of an un-promoted span, which should be rare in practice.

@ghost ghost added the in-pr There is an active PR which will close this issue when it is merged label Jun 30, 2023
@ghost ghost added in-pr There is an active PR which will close this issue when it is merged and removed in-pr There is an active PR which will close this issue when it is merged labels Sep 25, 2023
@ghost ghost removed the in-pr There is an active PR which will close this issue when it is merged label Oct 10, 2023
@ghost ghost locked as resolved and limited conversation to collaborators Nov 9, 2023
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
area-CodeGen-coreclr CLR JIT compiler in src/coreclr/src/jit and related components such as SuperPMI help wanted [up-for-grabs] Good issue for external contributors
Projects
None yet
Development

Successfully merging a pull request may close this issue.

3 participants