-
Notifications
You must be signed in to change notification settings - Fork 4.7k
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
Reduce asm size of CollectionsMarshal.AsSpan #96773
Conversation
Using the span constructor brings with it some branches and code we don't need due to knowledge of `List<T>`'s implementation, in particular that a) it always has an array (even if empty), and b) its array is invariant. Erroneous use of AsSpan while the list is being mutated could have also led to an ArgumentOutOfRangeException, which doesn't make much sense in this context. I've changed it to be an InvalidOperationException about the invalid concurrent use.
Tagging subscribers to this area: @dotnet/interop-contrib Issue DetailsUsing the span constructor brings with it some branches and code we don't need due to knowledge of Erroneous use of AsSpan while the list is being mutated could have also led to an ArgumentOutOfRangeException from the span's constructor, which doesn't make much sense in this context. I've changed it to be an InvalidOperationException about the invalid concurrent use.
using BenchmarkDotNet.Attributes;
using BenchmarkDotNet.Running;
using System.Runtime.InteropServices;
BenchmarkSwitcher.FromAssembly(typeof(Tests).Assembly).Run(args);
[DisassemblyDiagnoser]
public class Tests
{
private readonly List<string> _list = new() { "a", "b", "c" };
[Benchmark]
public Span<string> AsSpan() => CollectionsMarshal.AsSpan(_list);
[Benchmark]
public int Sum()
{
int total = 0;
foreach (string s in CollectionsMarshal.AsSpan(_list))
{
total += s.Length;
}
return total;
}
}
|
Why is the JIT not able to optimize the original code well? cc @dotnet/jit-contrib |
For starters I assume because it doesn't know that it can elide the null check on the array and elide the variance check on the array, since it's not privvy to the information about how the list was constructed. |
For the variance check, minimal repro: public class MyList<T>
{
public T[] arr;
}
bool Test(MyList<string> list) => list.arr.GetType() == typeof(string[]); codegen: ; Method Program:Foo(MyList`1[System.String]):ubyte:this (FullOpts)
G_M55733_IG01:
G_M55733_IG02:
mov rax, gword ptr [rdx+0x08]
mov rcx, 0xD1FFAB1E ; System.String[]
cmp qword ptr [rax], rcx
sete al
movzx rax, al
G_M55733_IG03:
ret
; Total bytes of code: 24 I bet it lost generic context somewhere during opts..
|
Even if that's fixed for string, though, I imagine the JIT would have a lot more trouble doing it for an arbitrary non-sealed reference type. |
Sure, but looks like it's should be a general goodness to improve this in JIT too if possible |
Agreed. Just commenting that this change still improves upon what I imagine the JIT will be able to safely do, unless we build knowledge of |
src/libraries/System.Private.CoreLib/src/System/Runtime/InteropServices/CollectionsMarshal.cs
Show resolved
Hide resolved
Yea, I agree. LGTM. |
Handled via #96794 |
src/libraries/System.Private.CoreLib/src/System/Runtime/InteropServices/CollectionsMarshal.cs
Outdated
Show resolved
Hide resolved
…pServices/CollectionsMarshal.cs
src/libraries/System.Private.CoreLib/src/System/Runtime/InteropServices/CollectionsMarshal.cs
Outdated
Show resolved
Hide resolved
…pServices/CollectionsMarshal.cs Co-authored-by: Jan Kotas <jkotas@microsoft.com>
* Reduce asm size of CollectionsMarshal.AsSpan Using the span constructor brings with it some branches and code we don't need due to knowledge of `List<T>`'s implementation, in particular that a) it always has an array (even if empty), and b) its array is invariant. Erroneous use of AsSpan while the list is being mutated could have also led to an ArgumentOutOfRangeException, which doesn't make much sense in this context. I've changed it to be an InvalidOperationException about the invalid concurrent use. * Update src/libraries/System.Private.CoreLib/src/System/Runtime/InteropServices/CollectionsMarshal.cs * Update src/libraries/System.Private.CoreLib/src/System/Runtime/InteropServices/CollectionsMarshal.cs Co-authored-by: Jan Kotas <jkotas@microsoft.com> --------- Co-authored-by: Jan Kotas <jkotas@microsoft.com>
Using the span constructor brings with it some branches and code we don't need due to knowledge of
List<T>
's implementation, in particular that a) it always has an array (even if empty), and b) its array is invariant.Erroneous use of AsSpan while the list is being mutated could have also led to an ArgumentOutOfRangeException from the span's constructor, which doesn't make much sense in this context. I've changed it to be an InvalidOperationException about the invalid concurrent use.