Share via


First-class Span Types

Note

This article is a feature specification. The specification serves as the design document for the feature. It includes proposed specification changes, along with information needed during the design and development of the feature. These articles are published until the proposed spec changes are finalized and incorporated in the current ECMA specification.

There may be some discrepancies between the feature specification and the completed implementation. Those differences are captured in the pertinent language design meeting (LDM) notes.

You can learn more about the process for adopting feature speclets into the C# language standard in the article on the specifications.

Champion issue: https://github.com/dotnet/csharplang/issues/8714

Summary

We introduce first-class support for Span<T> and ReadOnlySpan<T> in the language, including new implicit conversion types and consider them in more places, allowing more natural programming with these integral types.

Motivation

Since their introduction in C# 7.2, Span<T> and ReadOnlySpan<T> have worked their way into the language and base class library (BCL) in many key ways. This is great for developers, as their introduction improves performance without costing developer safety. However, the language has held these types at arm's length in a few key ways, which makes it hard to express the intent of APIs and leads to a significant amount of surface area duplication for new APIs. For example, the BCL has added a number of new tensor primitive APIs in .NET 9, but these APIs are all offered on ReadOnlySpan<T>. C# doesn't recognize the relationship between ReadOnlySpan<T>, Span<T>, and T[], so even though there are user-defined conversions between these types, they cannot be used for extension method receivers, cannot compose with other user-defined conversions, and don't help with all generic type inference scenarios. Users would need to use explicit conversions or type arguments, which means that IDE tooling is not guiding users to use these APIs, since nothing will indicate to the IDE that it is valid to pass these types after conversion. In order to provide maximum usability for this style of API, the BCL will have to define an entire set of Span<T> and T[] overloads, which is a lot of duplicate surface area to maintain for no real gain. This proposal seeks to address the problem by having the language more directly recognize these types and conversions.

For example, the BCL can add only one overload of any MemoryExtensions helper like:

int[] arr = [1, 2, 3];
Console.WriteLine(
    arr.StartsWith(1) // CS8773 in C# 13, permitted with this proposal
    );

public static class MemoryExtensions
{
    public static bool StartsWith<T>(this ReadOnlySpan<T> span, T value) where T : IEquatable<T> => span.Length != 0 && EqualityComparer<T>.Default.Equals(span[0], value);
}

Previously, Span and array overloads would be needed to make the extension method usable on Span/array-typed variables because user-defined conversions (which exist between Span/array/ReadOnlySpan) are not considered for extension receivers.

Detailed Design

The changes in this proposal will be tied to LangVersion >= 14.

Span conversions

We add a new type of implicit conversion to the list in §10.2.1, an implicit span conversion. This conversion is a conversion from type and is defined as follows:


An implicit span conversion permits array_types, System.Span<T>, System.ReadOnlySpan<T>, and string to be converted between each other as follows:

  • From any single-dimensional array_type with element type Ei to System.Span<Ei>
  • From any single-dimensional array_type with element type Ei to System.ReadOnlySpan<Ui>, provided that Ei is covariance-convertible (§18.2.3.3) to Ui
  • From System.Span<Ti> to System.ReadOnlySpan<Ui>, provided that Ti is covariance-convertible (§18.2.3.3) to Ui
  • From System.ReadOnlySpan<Ti> to System.ReadOnlySpan<Ui>, provided that Ti is covariance-convertible (§18.2.3.3) to Ui
  • From string to System.ReadOnlySpan<char>

Any Span/ReadOnlySpan types are considered applicable for the conversion if they are ref structs and they match by their fully-qualified name (LDM 2024-06-24).

We also add implicit span conversion to the list of standard implicit conversions (§10.4.2). This allows overload resolution to consider them when performing argument resolution, as in the previously-linked API proposal.

The explicit span conversions are the following:

  • All implicit span conversions.
  • From an array_type with element type Ti to System.Span<Ui> or System.ReadOnlySpan<Ui> provided an explicit reference conversion exists from Ti to Ui.

There is no standard explicit span conversion unlike other standard explicit conversions (§10.4.3) which always exist given the opposite standard implicit conversion.

User defined conversions

User-defined conversions are not considered when converting between types for which an implicit or an explicit span conversion exists.

The implicit span conversions are exempted from the rule that it is not possible to define a user-defined operator between types for which a non-user-defined conversion exists (§10.5.2 Permitted user-defined conversions). This is needed so the BCL can keep defining the existing Span conversion operators even when they switch to C# 14 (they are still needed for lower LangVersions and also because these operators are used in codegen of the new standard span conversions). But it can be viewed as an implementation detail (codegen and lower LangVersions are not part of the spec) and Roslyn violates this part of the spec anyway (this particular rule about user-defined conversions is not enforced).

Extension receiver

We also add implicit span conversion to the list of acceptable implicit conversions on the first parameter of an extension method when determining applicability (12.8.9.3) (change in bold):

An extension method Cᵢ.Mₑ is eligible if:

  • Cᵢ is a non-generic, non-nested class
  • The name of Mₑ is identifier
  • Mₑ is accessible and applicable when applied to the arguments as a static method as shown above
  • An implicit identity, reference or boxing , boxing, or span conversion exists from expr to the type of the first parameter of Mₑ. Span conversion is not considered when overload resolution is performed for a method group conversion.

Note that implicit span conversion is not considered for extension receiver in method group conversions (LDM 2024-07-15) which makes the following code continue working as opposed to resulting in a compile-time error CS1113: Extension method 'E.M<int>(Span<int>, int)' defined on value type 'Span<int>' cannot be used to create delegates:

using System;
using System.Collections.Generic;
Action<int> a = new int[0].M; // binds to M<int>(IEnumerable<int>, int)
static class E
{
    public static void M<T>(this Span<T> s, T x) => Console.Write(1);
    public static void M<T>(this IEnumerable<T> e, T x) => Console.Write(2);
}

As possible future work, we could consider removing this condition that span conversion is not considered for extension receiver in method group conversions and instead implement changes so the scenario like the one above would end up successfully calling the Span overload instead:

  • The compiler could emit a thunk that would take the array as the receiver and perform the span conversion inside (similarly to the user manually creating the delegate like x => new int[0].M(x)).
  • Value delegates if implemented could be able to take the Span as receiver directly.

Variance

The goal of the variance section in implicit span conversion is to replicate some amount of covariance for System.ReadOnlySpan<T>. Runtime changes would be required to fully implement variance through generics here (see ../csharp-13.0/ref-struct-interfaces.md for using ref struct types in generics), but we can allow a limited amount of covariance through use of a proposed .NET 9 API: https://github.com/dotnet/runtime/issues/96952. This will allow the language to treat System.ReadOnlySpan<T> as if the T was declared as out T in some scenarios. We do not, however, plumb this variant conversion through all variance scenarios, and do not add it to the definition of variance-convertible in §18.2.3.3. If in the future, we change the runtime to more deeply understand the variance here, we can take the minor breaking change to fully recognize it in the language.

Patterns

Note that when ref structs are used as a type in any pattern, only identity conversions are allowed:

class C<T> where T : allows ref struct
{
    void M1(T t) { if (t is T x) { } } // ok (T is T)
    void M2(R r) { if (r is R x) { } } // ok (R is R)
    void M3(T t) { if (t is R x) { } } // error (T is R)
    void M4(R r) { if (r is T x) { } } // error (R is T)
}
ref struct R { }

From the specification of the is-type operator (§12.12.12.1):

The result of the operation E is T [...] is a Boolean value indicating whether E is non-null and can successfully be converted to type T by a reference conversion, a boxing conversion, an unboxing conversion, a wrapping conversion, or an unwrapping conversion.

[...]

If T is a non-nullable value type, the result is true if D and T are the same type.

This behavior does not change with this feature, hence it will not be possible to write patterns for Span/ReadOnlySpan, although similar patterns are possible for arrays (including variance):

using System;

M1<object[]>(["0"]); // prints
M1<string[]>(["1"]); // prints

void M1<T>(T t)
{
    if (t is object[] r) Console.WriteLine(r[0]); // ok
}

void M2<T>(T t) where T : allows ref struct
{
    if (t is ReadOnlySpan<object> r) Console.WriteLine(r[0]); // error
}

Code generation

The conversions will always exist, regardless of whether any runtime helpers used to implement them are present (LDM 2024-05-13). If the helpers are not present, attempting to use the conversion will result in a compile-time error that a compiler-required member is missing.

The compiler expects to use the following helpers or equivalents to implement the conversions:

Conversion Helpers
array to Span static implicit operator Span<T>(T[]) (defined in Span<T>)
array to ReadOnlySpan static implicit operator ReadOnlySpan<T>(T[]) (defined in ReadOnlySpan<T>)
Span to ReadOnlySpan static implicit operator ReadOnlySpan<T>(Span<T>) (defined in Span<T>) and static ReadOnlySpan<T>.CastUp<TDerived>(ReadOnlySpan<TDerived>)
ReadOnlySpan to ReadOnlySpan static ReadOnlySpan<T>.CastUp<TDerived>(ReadOnlySpan<TDerived>)
string to ReadOnlySpan static ReadOnlySpan<char> MemoryExtensions.AsSpan(string)

Note that MemoryExtensions.AsSpan is used instead of the equivalent implicit operator defined on string. This means the codegen is different between LangVersions (the implicit operator is used in C# 13; the static method AsSpan is used in C# 14). On the other hand, the conversion can be emitted on .NET Framework (the AsSpan method exists there whereas the string operator does not).

The explicit array to (ReadOnly)Span conversion first converts explicitly from the source array to an array with the destination element type and then to (ReadOnly)Span via the same helper as an implicit conversion would use, i.e., the corresponding op_Implicit(T[]).

Better conversion from expression

Better conversion from expression (§12.6.4.5) is updated to prefer implicit span conversions. This is based on collection expressions overload resolution changes.

Given an implicit conversion C₁ that converts from an expression E to a type T₁, and an implicit conversion C₂ that converts from an expression E to a type T₂, C₁ is a better conversion than C₂ if one of the following holds:

  • E is a collection expression, and C₁ is a better collection conversion from expression than C₂
  • E is not a collection expression and one of the following holds:
    • E exactly matches T₁ and E does not exactly match T₂
    • E exactly matches neither of T₁ and T₂, and C₁ is an implicit span conversion and C₂ is not an implicit span conversion
    • E exactly matches both or neither of T₁ and T₂, both or neither of C₁ and C₂ are an implicit span conversion, and T₁ is a better conversion target than T₂
  • E is a method group, T₁ is compatible with the single best method from the method group for conversion C₁, and T₂ is not compatible with the single best method from the method group for conversion C₂

Better conversion target

Better conversion target (§12.6.4.7) is updated to prefer ReadOnlySpan<T> over Span<T>.

Given two types T₁ and T₂, T₁ is a better conversion target than T₂ if one of the following holds:

  • T₁ is System.ReadOnlySpan<E₁>, T₂ is System.Span<E₂>, and an identity conversion from E₁ to E₂ exists
  • T₁ is System.ReadOnlySpan<E₁>, T₂ is System.ReadOnlySpan<E₂>, and an implicit conversion from T₁ to T₂ exists and no implicit conversion from T₂ to T₁ exists
  • At least one of T₁ or T₂ is not System.ReadOnlySpan<Eᵢ> and is not System.Span<Eᵢ>, and an implicit conversion from T₁ to T₂ exists and no implicit conversion from T₂ to T₁ exists
  • ...

Design meetings:

Betterness remarks

The better conversion from expression rule should ensure that whenever an overload becomes applicable due to the new span conversions, any potential ambiguity with another overload is avoided because the newly-applicable overload is preferred.

Without this rule, the following code that successfully compiled in C# 13 would result in an ambiguity error in C# 14 because of the new standard implicit conversion from array to ReadOnlySpan applicable to an extension method receiver:

using System;
using System.Collections.Generic;

var a = new int[] { 1, 2, 3 };
a.M();

static class E
{
    public static void M(this IEnumerable<int> x) { }
    public static void M(this ReadOnlySpan<int> x) { }
}

The rule also allows introducing new APIs that would previously result in ambiguities, for example:

using System;
using System.Collections.Generic;

C.M(new int[] { 1, 2, 3 }); // would be ambiguous before

static class C
{
    public static void M(IEnumerable<int> x) { }
    public static void M(ReadOnlySpan<int> x) { } // can be added now
}

Warning

Because the betterness rule is defined for the span conversions which only exist in LangVersion >= 14, API authors cannot add such new overloads if they want to keep supporting users on LangVersion <= 13. For example, if .NET 9 BCL introduces such overloads, users that upgrade to net9.0 TFM but stay on lower LangVersion will get ambiguity errors for existing code. See also an open question below.

Type inference

We update the type inferences section of the specification as follows (changes in bold).

12.6.3.9 Exact inferences

An exact inference from a type U to a type V is made as follows:

  • If V is one of the unfixed Xᵢ then U is added to the set of exact bounds for Xᵢ.
  • Otherwise, sets V₁...Vₑ and U₁...Uₑ are determined by checking if any of the following cases apply:
    • V is an array type V₁[...] and U is an array type U₁[...] of the same rank
    • V is a Span<V₁> and U is an array type U₁[] or a Span<U₁>
    • V is a ReadOnlySpan<V₁> and U is an array type U₁[] or a Span<U₁> or ReadOnlySpan<U₁>
    • V is the type V₁? and U is the type U₁
    • V is a constructed type C<V₁...Vₑ> and U is a constructed type C<U₁...Uₑ>
      If any of these cases apply then an exact inference is made from each Uᵢ to the corresponding Vᵢ.
  • Otherwise, no inferences are made.

12.6.3.10 Lower-bound inferences

A lower-bound inference from a type U to a type V is made as follows:

  • If V is one of the unfixed Xᵢ then U is added to the set of lower bounds for Xᵢ.
  • Otherwise, if V is the type V₁? and U is the type U₁? then a lower bound inference is made from U₁ to V₁.
  • Otherwise, sets U₁...Uₑ and V₁...Vₑ are determined by checking if any of the following cases apply:
    • V is an array type V₁[...] and U is an array type U₁[...]of the same rank
    • V is a Span<V₁> and U is an array type U₁[] or a Span<U₁>
    • V is a ReadOnlySpan<V₁> and U is an array type U₁[] or a Span<U₁> or ReadOnlySpan<U₁>
    • V is one of IEnumerable<V₁>, ICollection<V₁>, IReadOnlyList<V₁>>, IReadOnlyCollection<V₁> or IList<V₁> and U is a single-dimensional array type U₁[]
    • V is a constructed class, struct, interface or delegate type C<V₁...Vₑ> and there is a unique type C<U₁...Uₑ> such that U (or, if U is a type parameter, its effective base class or any member of its effective interface set) is identical to, inherits from (directly or indirectly), or implements (directly or indirectly) C<U₁...Uₑ>.
    • (The “uniqueness” restriction means that in the case interface C<T>{} class U: C<X>, C<Y>{}, then no inference is made when inferring from U to C<T> because U₁ could be X or Y.)
      If any of these cases apply then an inference is made from each Uᵢ to the corresponding Vᵢ as follows:
    • If Uᵢ is not known to be a reference type then an exact inference is made
    • Otherwise, if U is an array type then a lower-bound inference is made inference depends on the type of V:
      • If V is a Span<Vᵢ>, then an exact inference is made
      • If V is an array type or a ReadOnlySpan<Vᵢ>, then a lower-bound inference is made
    • Otherwise, if U is a Span<Uᵢ> then inference depends on the type of V:
      • If V is a Span<Vᵢ>, then an exact inference is made
      • If V is a ReadOnlySpan<Vᵢ>, then a lower-bound inference is made
    • Otherwise, if U is a ReadOnlySpan<Uᵢ> and V is a ReadOnlySpan<Vᵢ> a lower-bound inference is made:
    • Otherwise, if V is C<V₁...Vₑ> then inference depends on the i-th type parameter of C:
      • If it is covariant then a lower-bound inference is made.
      • If it is contravariant then an upper-bound inference is made.
      • If it is invariant then an exact inference is made.
  • Otherwise, no inferences are made.

There are no rules for upper-bound inference because it would not be possible to hit them. Type inference never starts as upper-bound, it would have to go through a lower-bound inference and a contravariant type parameter. Because of the rule "if Uᵢ is not known to be a reference type then an exact inference is made," the source type argument could not be Span/ReadOnlySpan (those cannot be reference types). However, the upper-bound span inference would only apply if the source type were a Span/ReadOnlySpan, since it would have rules like:

  • U is a Span<U₁> and V is an array type V₁[] or a Span<V₁>
  • U is a ReadOnlySpan<U₁> and V is an array type V₁[] or a Span<V₁> or ReadOnlySpan<V₁>

Breaking changes

As any proposal that changes conversions of existing scenarios, this proposal does introduce some new breaking changes. Here's a few examples:

Calling Reverse on an array

Calling x.Reverse() where x is an instance of type T[] would previously bind to IEnumerable<T> Enumerable.Reverse<T>(this IEnumerable<T>), whereas now it binds to void MemoryExtensions.Reverse<T>(this Span<T>). Unfortunately these APIs are incompatible (the latter does the reversal in-place and returns void).

.NET 10 mitigates this by adding an array-specific overload IEnumerable<T> Reverse<T>(this T[]), see https://github.com/dotnet/runtime/issues/107723.

void M(int[] a)
{
    foreach (var x in a.Reverse()) { } // fine previously, an error now (`Reverse` returns `void`)
    foreach (var x in Enumerable.Reverse(a)) { } // workaround
}

See also:

Design meeting: https://github.com/dotnet/csharplang/blob/main/meetings/2024/LDM-2024-09-11.md#reverse

Ambiguities

The following examples previously failed type inference for the Span overload, but now type inference from array to Span succeeds, hence these are ambiguous. To work around this, users can use .AsSpan() or API authors can use OverloadResolutionPriorityAttribute.

var x = new long[] { 1 };
Assert.Equal([2], x); // previously Assert.Equal<T>(T[], T[]), now ambiguous with Assert.Equal<T>(ReadOnlySpan<T>, Span<T>)
Assert.Equal([2], x.AsSpan()); // workaround
var x = new int[] { 1, 2 };
var s = new ArraySegment<int>(x, 1, 1);
Assert.Equal(x, s); // previously Assert.Equal<T>(T, T), now ambiguous with Assert.Equal<T>(Span<T>, Span<T>)
Assert.Equal(x.AsSpan(), s); // workaround

xUnit is adding more overloads to mitigate this: https://github.com/xunit/xunit/discussions/3021.

Design meeting: https://github.com/dotnet/csharplang/blob/main/meetings/2024/LDM-2024-09-11.md#new-ambiguities

Covariant arrays

Overloads taking IEnumerable<T> worked on covariant arrays, but overloads taking Span<T> (which we now prefer) don't, because the span conversion throws an ArrayTypeMismatchException for covariant arrays. Arguably, the Span<T> overload should not exist, it should take ReadOnlySpan<T> instead. To work around this, users can use .AsEnumerable(), or API authors can use OverloadResolutionPriorityAttribute or add ReadOnlySpan<T> overload which is preferred due to the betterness rule.

string[] s = new[] { "a" };
object[] o = s;

C.R(o); // wrote 1 previously, now crashes in Span<T> constructor with ArrayTypeMismatchException
C.R(o.AsEnumerable()); // workaround

static class C
{
    public static void R<T>(IEnumerable<T> e) => Console.Write(1);
    public static void R<T>(Span<T> s) => Console.Write(2);
    // another workaround:
    public static void R<T>(ReadOnlySpan<T> s) => Console.Write(3);
}

Design meeting: https://github.com/dotnet/csharplang/blob/main/meetings/2024/LDM-2024-09-11.md#covariant-arrays

Preferring ReadOnlySpan over Span

The betterness rule causes preference of ReadOnlySpan overloads over Span overloads to avoid ArrayTypeMismatchExceptions in covariant array scenarios. That can lead to compilation breaks in some scenarios, for example when the overloads differ by their return type:

double[] x = new double[0];
Span<ulong> y = MemoryMarshal.Cast<double, ulong>(x); // previously worked, now a compilation error (returns ReadOnlySpan, not Span)
Span<ulong> z = MemoryMarshal.Cast<double, ulong>(x.AsSpan()); // workaround

static class MemoryMarshal
{
    public static ReadOnlySpan<TTo> Cast<TFrom, TTo>(ReadOnlySpan<TFrom> span) => default;
    public static Span<TTo> Cast<TFrom, TTo>(Span<TFrom> span) => default;
}

See https://github.com/dotnet/roslyn/issues/76443.

Expression trees

Overloads taking spans like MemoryExtensions.Contains are preferred over classic overloads like Enumerable.Contains, even inside expression trees - but ref structs are not supported by the interpreter engine:

Expression<Func<int[], int, bool>> exp = (array, num) => array.Contains(num);
exp.Compile(preferInterpretation: true); // fails at runtime in C# 14

Expression<Func<int[], int, bool>> exp2 = (array, num) => Enumerable.Contains(array, num); // workaround
exp2.Compile(preferInterpretation: true); // ok

Similarly, translation engines like LINQ-to-SQL need to react to this if their tree visitors expect Enumerable.Contains because they will encounter MemoryExtensions.Contains instead.

See also:

Design meetings:

User-defined conversions through inheritance

By adding implicit span conversions to the list of standard implicit conversions, we can potentially change behavior when user-defined conversions are involved in a type hierarchy. This example shows that change, in comparison to an integer scenario that already behaves as the new C# 14 behavior will.

Span<string> span = [];
var d = new Derived();
d.M(span); // Base today, Derived tomorrow
int i = 1;
d.M(i); // Derived today, demonstrates new behavior

class Base
{
    public void M(Span<string> s)
    {
        Console.WriteLine("Base");
    }

    public void M(int i)
    {
        Console.WriteLine("Base");
    }
}

class Derived : Base
{
    public static implicit operator Derived(ReadOnlySpan<string> r) => new Derived();
    public static implicit operator Derived(long l) => new Derived();

    public void M(Derived s)
    {
        Console.WriteLine("Derived");
    }
}

See also: https://github.com/dotnet/roslyn/issues/78314

Extension method lookup

By allowing implicit span conversions in extension method lookup, we can potentially change what extension method is resolved by overload resolution.

namespace N1
{
    using N2;

    public class C
    {
        public static void M()
        {
            Span<string> span = new string[0];
            span.Test(); // Prints N2 today, N1 tomorrow
        }
    }

    public static class N1Ext
    {
        public static void Test(this ReadOnlySpan<string> span)
        {
            Console.WriteLine("N1");
        }
    }
}

namespace N2
{
    public static class N2Ext
    {
        public static void Test(this Span<string> span)
        {
            Console.WriteLine("N2");
        }
    }
}

Open questions

Unrestricted betterness rule

Should we make the betterness rule unconditional on LangVersion? That would allow API authors to add new Span APIs where IEnumerable equivalents exist without breaking users on older LangVersions or other compilers or languages (e.g., VB). However, that would mean users could get different behavior after updating the toolset (without changing LangVersion or TargetFramework):

  • Compiler could choose different overloads (technically a breaking change, but hopefully those overloads would have equivalent behavior).
  • Other breaks could arise, unknown at this time.

Note that OverloadResolutionPriorityAttribute cannot fully solve this because it's also ignored on older LangVersions. However, it should be possible to use it to avoid ambiguities from VB where the attribute should be recognized.

Ignoring more user-defined conversions

We defined a set of type pairs for which there are language-defined implicit and explicit span conversions. Whenever a language-defined span conversion exists from T1 to T2, any user-defined conversion from T1 to T2 is ignored (regardless of the span and user-defined conversion being implicit or explicit).

Note that this includes all the conditions, so for example there is no span conversion from Span<object> to ReadOnlySpan<string> (there is a span conversion from Span<T> to ReadOnlySpan<U> but it must hold that T : U), hence a user-defined conversion would be considered between those types if it existed (that would have to be a specialized conversion like Span<T> to ReadOnlySpan<string> because conversion operators cannot have generic parameters).

Should we ignore user-defined conversions also between other combinations of array/Span/ReadOnlySpan/string types where no corresponding language-defined span conversion exists? For example, if there is a user-defined conversion from ReadOnlySpan<T> to Span<T>, should we ignore it?

Spec possibilities to consider:

  1. Whenever a span conversion exists from T1 to T2, ignore any user-defined conversion from T1 to T2 or from T2 to T1.

  2. User-defined conversions are not considered when converting between

    • any single-dimensional array_type and System.Span<T>/System.ReadOnlySpan<T>,
    • any combination of System.Span<T>/System.ReadOnlySpan<T>,
    • string and System.ReadOnlySpan<char>.
  3. Like above but replacing the last bullet point with:
    • string and System.Span<char>/System.ReadOnlySpan<char>.
  4. Like above but replacing the last bullet point with:
    • string and System.Span<T>/System.ReadOnlySpan<T>.

Technically, the spec disallows some of these user-defined conversions to be even defined: it is not possible to define a user-defined operator between types for which a non-user-defined conversion exists (§10.5.2). But Roslyn intentionally violates this part of the spec. And some conversions like between Span and string are allowed anyway (no language-defined conversion between these types exist).

Nevertheless, alternatively to just ignoring the conversions, we could disallow them to be defined at all and perhaps break out of the spec violation at least for these new span conversions, i.e., change Roslyn to actually report a compile-time error if these conversions are defined (likely except those already defined by the BCL).

Alternatives

Keep things as they are.