Breaking changes in Roslyn after .NET 8.0.100 through .NET 9.0.100
Article
This document lists known breaking changes in Roslyn after .NET 8 general release (.NET SDK version 8.0.100) through .NET 9 general release (.NET SDK version 9.0.100).
InlineArray attribute on a record struct type is no longer allowed.
Introduced in Visual Studio 2022 version 17.11
cs
[System.Runtime.CompilerServices.InlineArray(10)] // error CS9259: Attribute 'System.Runtime.CompilerServices.InlineArray' cannot be applied to a record struct.recordstructBuffer1()
{
privateint _element0;
}
[System.Runtime.CompilerServices.InlineArray(10)] // error CS9259: Attribute 'System.Runtime.CompilerServices.InlineArray' cannot be applied to a record struct.recordstructBuffer2(int p1)
{
}
Iterators introduce safe context in C# 13 and newer
Introduced in Visual Studio 2022 version 17.11
Although the language spec states that iterators introduce a safe context, Roslyn does not implement that in C# 12 and lower.
This will change in C# 13 as part of a feature which allows unsafe code in iterators.
The change does not break normal scenarios as it was disallowed to use unsafe constructs directly in iterators anyway.
However, it can break scenarios where an unsafe context was previously inherited into nested local functions, for example:
cs
unsafeclassC// unsafe context
{
System.Collections.Generic.IEnumerable<int> M() // an iterator
{
yieldreturn1;
local();
voidlocal()
{
int* p = null; // allowed in C# 12; error in C# 13
}
}
}
You can work around the break simply by adding the unsafe modifier to the local function.
Collection expression breaking changes with overload resolution in C# 13 and newer
Introduced in Visual Studio 2022 Version 17.12 and newer when using C# 13+
There are a few changes in collection expression binding in C# 13. Most of these are turning ambiguities into successful compilations,
but a couple are breaking changes that either result in a new compilation error, or are a behavior breaking change. They are detailed
below.
Empty collection expressions no longer use whether an API is a span to tiebreak on overloads
When an empty collection expression is provided to an overloaded method, and there isn't a clear element type, we no longer use whether
an API takes a ReadOnlySpan<T> or a Span<T> to decide whether to prefer that API. For example:
In C# 13, we prefer an exact element type match, looking at conversions from expressions. This can result in a behavior change when involving
constants:
cs
classC
{
staticvoidM1(ReadOnlySpan<byte> ros) {}
staticvoidM1(Span<int> s) {}
staticvoidM2(ReadOnlySpan<string> ros) {}
staticvoidM2(Span<CustomInterpolatedStringHandler> ros) {}
staticvoidMain()
{
M1([1]); // C.M(ReadOnlySpan<byte>) in C# 12, C.M(Span<int>) in C# 13
M2([$"{1}"]); // C.M(ReadOnlySpan<string>) in C# 12, C.M(Span<CustomInterpolatedStringHandler>) in C# 13
}
}
Declaration of indexers in absence of proper declaration of DefaultMemberAttribute is no longer allowed.
Introduced in Visual Studio 2022 version 17.13
cs
publicinterfaceI1
{
public I1 this[I1 args] { get; } // error CS0656: Missing compiler required member 'System.Reflection.DefaultMemberAttribute..ctor'
}
Default and params parameters are considered in method group natural type
Introduced in Visual Studio 2022 version 17.13
Previously the compiler unexpectedly
inferred different delegate type depending on the order of candidates in source
when default parameter values or params arrays were used. Now an ambiguity error is emitted.
cs
using System;
classProgram
{
staticvoidMain()
{
var x1 = new Program().Test1; // previously Action<long[]> - now errorvar x2 = new Program().Test2; // previously anonymous void delegate(params long[]) - now error
x1();
x2();
}
}
staticclassE
{
staticpublicvoidTest1(this Program p, long[] a) => Console.Write(a.Length);
staticpublicvoidTest1(thisobject p, paramslong[] a) => Console.Write(a.Length);
staticpublicvoidTest2(thisobject p, paramslong[] a) => Console.Write(a.Length);
staticpublicvoidTest2(this Program p, long[] a) => Console.Write(a.Length);
}
Also in LangVersion=12 or lower, params modifier must match across all methods to infer a unique delegate signature.
Note that this does not affect LangVersion=13 and later because of a different delegate inference algorithm.
cs
var d = new C().M; // previously inferred Action<int[]> - now error CS8917: the delegate type could not be inferredstaticclassE
{
publicstaticvoidM(this C c, paramsint[] x) { }
}
classC
{
publicvoidM(int[] x) { }
}
A workaround is to use explicit delegate types instead of relying on var inference in those cases.
dotnet_style_require_accessibility_modifiers now consistently applies to interface members
Prior to this change, the analyzer for dotnet_style_require_accessibility_modifiers would simply ignore interface
members. This was because C# initially disallowed modifiers for interface members entirely, having them always
be public.
Later versions of the language relaxed this restriction, allowing users to provide accessibility modifiers on
interface members, including a redundant public modifier.
The analyzer was updated to now enforce the value for this option on interface members as well. The meaning
of the value is as follows:
never. The analyzer does no analysis. Redundant modifiers are allowed on all members.
always. Redundant modifiers are always required on all members (including interface members). For example:
a private modifier on a class member, and a public modifier on an interface member. This is the option to
use if you feel that all members no matter what should state their accessibility explicitly.
for_non_interface_members. Redundant modifiers are required on all members that are not part of an interface,
but disallowed for interface members. For example: private will be required on private class members. However,
a public interface member will not be allowed to have redundant public modifiers. This matches the standard
modifier approach present prior to the language allowing modifiers on interface members.
omit_if_default. Redundant modifiers are disallowed. For example a private class member will be disallowed from
using private, and a public interface member will be disallowed from using public. This is the option to use
if you feel that restating the accessibility when it matches what the language chooses by default is redundant and
should be disallowed.
Collaborate with us on GitHub
The source for this content can be found on GitHub, where you can also create and review issues and pull requests. For more information, see our contributor guide.
Roslyn breaking changes feedback
Roslyn breaking changes is an open source project. Select a link to provide feedback:
In this module, you explore advanced concepts of interfaces in C#. You learn how to implement explicit interface members, combine multiple interfaces, and reduce code dependencies using interfaces.