diff --git a/docs/monomorphic-type-bindings-for-generics.md b/docs/monomorphic-type-bindings-for-generics.md new file mode 100644 index 0000000..be8299a --- /dev/null +++ b/docs/monomorphic-type-bindings-for-generics.md @@ -0,0 +1,56 @@ +# Monomorphic Type Bindings for Generics + +## Summary + +Introduce a way to opt into monomorphic (greedy) type inference for generics, preventing them from widening into unions when instantiated with different types. + +## Motivation + +Currently, Luau's type solver allows generic parameters to widen into unions when they are instantiated with multiple types. For example: + +```luau +function test(a: T, b: T): T + return a +end + +local result = test(1, "string") -- inferred type: 1 | "string" +``` + +This behaviour can be useful in some cases but is undesirable when a function is intended to enforce strict monomorphism. A monomorphic binding would prevent `T` from being instantiated with different types and instead produce a type error if `b` does not match `a`. + +```luau +function test(a: T, b: T) +end + +test(1, "string") -- Type error: Expected `number`, got `string` +``` + +## Design + +This RFC proposes adding some symbol next to each type variable which is a clear opt-in. + +### New Syntax + +The `T!` syntax would indicate a that the type binding for `T` is monomorphic. + +### Type Checking Rules + +1. When a generic parameter is marked as `T!`: + - The first instantiation of `T` determines its type + - Any subsequent use of `T` in the same context must match this type exactly + - If a different type is encountered, a **type error** is raised + - `T` will not expand into a union + +2. **Behavior in Unions** + - A function or type with `T!` cannot instantiate `T` with a union + - If `T` is already a union, it must remain a union as new types cannot be added to it + +## Drawbacks + +- Introduces a new syntax modifier (`T!`), which may lead to a symbol soup but it doesn't seem too shabby next to `T?`. + +## Alternatives + +- **Type Function Constraint**: Provide a `types.monomorphic` function in user-defined type functions to enforce monomorphism dynamically. But this would probably require some way to propagate an `*error-type*` to the user +- **Keywords**: Something like `` or `` should also be considered if we want to reduce symbols +- **Field-specific bindings**: Flip the relationship from being declared per-type-parameter to be set per-function-parameter such as `function test(a: T, b: T!)` which would imply allow both inference behaviours of generics in an uniform syntax