-
Notifications
You must be signed in to change notification settings - Fork 790
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
FSharp.UMX no longer compiles on .NET 7 #12881
Comments
So the problem here is that there is specific logic for computing the apparent interfaces of [<MeasureAnnotatedAbbreviation>] type Guid<[<Measure>] 'm> = System.Guid What interfaces does The problem is, this is only done for a fixed, known set of interface - because it is baking in a set of assumptions about how these interfaces behave with regard to the unit annotations. And in .NET 7, there are a lot of important new interfaces appearing on these primitive types that get used with .NET 6: public readonly struct Guid : IComparable, IComparable<Guid>, IEquatable<Guid>, ISpanFormattable .NET 7: public readonly struct Guid : IComparable<Guid>, IComparisonOperators<Guid,Guid>, IEqualityOperators<Guid,Guid>, IEquatable<Guid>, IParseable<Guid>, ISpanFormattable, ISpanParseable<Guid> Here |
Note the list of interfaces appearing is large, e.g. for System.Double: public readonly struct Double :
IAdditionOperators<double,double,double>, IAdditiveIdentity<double,double>,
IBinaryFloatingPoint<double>, IBinaryNumber<double>, IBitwiseOperators<double,double,double>,
IComparable<double>, IComparisonOperators<double,double>, IConvertible,
IDecrementOperators<double>, IDivisionOperators<double,double,double>,
IEqualityOperators<double,double>, IEquatable<double>, IFloatingPoint<double>,
IIncrementOperators<double>, IMinMaxValue<double>,
IModulusOperators<double,double,double>,
IMultiplicativeIdentity<double,double>, IMultiplyOperators<double,double,double>,
INumber<double>, IParseable<double>, ISignedNumber<double>,
ISpanParseable<double>, ISubtractionOperators<double,double,double>,
IUnaryNegationOperators<double,double>, IUnaryPlusOperators<double,double> I will admit I find that interface list overkill and I fear for the simplicity of the whole of .NET, but hey. I also fear a little for the performance of the F# compiler as we don't generally expect to see quite so many interfaces of such complexity on primitive types. I wonder if we will start to hit more bottlenecks in interface processing code. Anyway, the specific technical problem of what to do about these interfaces for "unitized" types is significant and problematic and something we need to solve. For example the unitized
is fine but
would be wrong, it should be minimally
Even that - while sound - is not ideal - as the units should in theory be allowed to be different for the two operators, so it should really be
But where is @tannergooding You might like to take a look at this. Also it would be good to get an understanding for how many further changes are in the works for these fundamental interface types, as it looks like the F# compiler will need to be taught some understanding of them to get the unit of measure feature to play out as soundly and nicely as possible. |
.NET is largely statically/strongly typed and needs to support a broad range of scenarios and languages. Many of these interfaces are built for extensibility and reusability so that everything can have the right amount of flexibility and you can find the relevant information in the type system. The hierarchy exists in order to support a broad range of requirements as you need to be able to support and represent things like
The runtime was able to workaround the perf implications here and essentially mitigate the cost altogether. One of the ways it did this was to specialize the primitive types. I believe F# should be able to achieve the same and likewise handle the potential perf implications of types implementing such complex hierarchies.
The next API review is tomorrow and it will result in some additional changes around the name/shape of certain APIs, as well as the shuffling about of various APIs to better fit the feedback we've gotten from the community.
This isn't quite right. The hierarchy is:
That is For basically all of the interfaces, there is largely a single "root" interface implemented by these types and that pulls in "everything else". In the case of most of the integer types, this is |
The complexity is well-motivated, of course, as complexity always is. The question here is the impact of complexification. Anyway, this is not the place to discuss this - we should focus on the specific bug here, plus the general question of "what's going to go wrong when F# sees this new metadata". But as a general point: I fear for a platform that doesn't have simplicity as the primary core virtue - one valued more highly than extensibility, reusability, flexibility etc. And what's being added here is most definitely not simple.
I'd imagine that's right, though beware there's a risk we're going to discover this through a series of perf regressions only experienced in particular scenarios, or through general reduction in type checking performance or IDE tooling. There's some risk there will be a quadratic or worse performance reduction in type inference, again only detected once starting to use .NET 7 for real. Note no one is specifically testing or monitoring for such performance regressions, they may just start happening once F# starts being used against .NET 7 metadata, and perhaps only in corner cases. It's not certain this will happen, it's just a risk. If we increase the complexity of any metadata from say 10 to 100 interfaces (taking into account the full hierarchy) it's pretty certain to cause some kinds of problems. |
Here is a .NET 6 repro for the specific problem with F# user-defined unit-of-measure-annotated types where the underlying type supports an interface derived from IComparable or IEquality. I will fix this to unblock. open System
type IDerivedComparable<'T> =
inherit IComparable<'T>
type Prim() =
interface IDerivedComparable<Prim> with
member x.CompareTo(y) = 0
[<MeasureAnnotatedAbbreviation>]
type Prim<[<Measure>] 'm> = Prim
// Check that Prim<'m> supports the unit-annotated IComparable interface
let f1 (x: Prim<'m>) = (x :> IComparable<Prim<'m>>)
let f2 (x: Prim<'m>) = (x :> IDerivedComparable<Prim<'m>>)
|
The ix for the immediate issue is in #12892 Looking beyond, my assessment is that the .NET 7 hierarchy just doesn't jive with units of measure (e.g. would need a complete redesign were C# ever to support units of measure in its type system). In practice I believe this means .NET and C# will never support units of measure in the type system (except perhaps via some kind of separate code analysis). As an example, consider some of the interfaces appearing in the .NET 7 mega-hierarchy:
Why doesn't F# hit this problem today with its generic math capabilities? It's because F# uses structural SRTP constraints and these are able to capture the patterns involved in both non-unitized and unitized types. For example, in F# today is is possible to generalize arithmetic and then later satisfy the SRTP constraints that arise with either unitized or non-unitized types: let inline someGenericMath x y = x * y + x * y
let g0 (x: float) (y: float) = someGenericMath x y
let g1 (x: float<'u>) (y: float<'v>) = someGenericMath x y
let g2 (x: int<'v>) (y: int<'v>) = someGenericMath x y with inferred types: val inline someGenericMath:
x: ^a -> y: ^b -> ^d
when ( ^a or ^b) : (static member ( * ) : ^a * ^b -> ^c) and
^c: (static member (+) : ^c * ^c -> ^d)
val g0: x: float -> y: float -> float
val g1: x: float<'u> -> y: float<'v> -> float<'u 'v>
val g2: x: int<'v> -> y: int<'v> -> int<'v ^ 2> Here SRTP constraints arise from the use-site of each operator |
Notably we do have C# tests (and in some cases IL tests) covering the impact of startup, devirtualization, casting, type checks, etc. These generally live in https://github.com/dotnet/performance and we do weekly triage of these for Windows, Linux, and MacOS and covering x86/x64 and Arm32/Arm64. If there are F# scenarios outside what we are already testing, then we should definitely look at adding those as well so we can test the impact over time and likewise prevent regressions for important F# specific scenarios (whether specific to
For certain and we specifically order interface declarations to reduce this cost for the most common usage scenarios. This is one of the reasons why Various common coding patterns also help reduce the costs. Constrained generics can mitigate checks, generics over value types will mitigate checks (the JIT specializes in this case), and basically any other place where the type can be statically determined will likewise have the cost reduced or completely mitigated (JIT time constant). The remaining cases are largely where you have generics over a reference type or some base type (including We'll continue carefully monitoring the impact of these things and common patterns users are utilizing around here to try and ensure that these costs stay down and aren't negatively impactful. |
My concern is compilation performance (and in particular type inference and name resolution performance both in compiler and in IDE, i.e. any logic in toolchains that involves walking the hierarchy of interfaces looking at metadata along the way). For example a compiler might search the hierarchy for, say IDisposable, or IEnumerable, or might search for name resolutions. All of this will be at least slightly impacted in some scenarios - more information, more cost - but it's unclear to me if there's any critical case where this occurs on a main path for compilation. The performance of generated code is not a concern I have, I'm sure you'll look after that. |
If there are specific suggestions on tweaks we could make to the surface area to better support F# or F# specific features, then I'm all ears. We need to support a range of concepts including those that are outside "standard arithmetic". This includes concepts like This representation ensures that a wide variety of languages are able to utilize the relevant metadata and features and ensures optimizations can happen at any stage of the development process (whether manually done by the user, handled via compiler optimizations, or handled via JIT/AOT optimizations in processing the IL). It also ensures that the full breadth of the .NET stack can be represented as can concepts that programmers need to deal with that mathematicians may not (this is why we don't represent or support concepts such as commutativity, associativity, distributivity, rings, groups, fields, etc). |
I don't have any concrete suggestions - Generic math is basically a bottomless quagmire of complexity as more and more dimensions of the problem are considered - and what you have is already at the upper limit of any reasonable complexity boundary - adding unitization would be many bridges too far. As you've mentioned, .NET 7 uses explicit-declaration-of-traits-via-interfaces (rather than, say, implicit trait instances arising from the declarations themselves, as in SRTP), and expresses those declarations using the existing C# and IL interfaces. It is just not possible to properly support unitized types within those limitations (without adding higher-kinded types at very least - and all the complexity that comes with using that). For example of why higher-kinded typed would be needed - interface IMultiplyOps<UnitizableType1, UnitizableType2, UnitizableType3>
{
UnitizableType3<Unit1*Unit2> op_Multiply<Unit1, Unit2>(UnitizableType1<Unit1> x, UnitizableType2<Unit2> y)
}
struct Double<Unit>: IMultiplyOps<Double<_>, Double<_>, Double<_>>
{
Double<Unit1*Unit2> op_Multiply<Unit1, Unit2>(Double<Unit1> x, Double<Unit2> y)
...
} Note
Anyway none of this is expressible within .NET or C# or F#. I think on the whole we just won't worry about this. F# will still play nicely enough with the non-unitized hierarchy you have and the type inference will work well. We'll just make it clear that there are limitations when combining the feature with units of measure. |
Repro steps
Install .NET SDK 7.0.100-preview.4.22175.1 or newer and run this in fsi
Expected behavior
Compiles.
Actual behavior
Known workarounds
Downgrade to 7.0.100-preview.2.22103.2.
The text was updated successfully, but these errors were encountered: