Skip to content

Commit 38af45a

Browse files
Update SIPs state
1 parent 0d8a969 commit 38af45a

7 files changed

+277
-23
lines changed

_sips/sips/alternative-bind-patterns.md

-7
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
---
2+
title: SIP-62 - For comprehension improvements
3+
status: submitted
4+
pull-request-number: 79
5+
stage: pre-sip
6+
7+
---

_sips/sips/match-types-spec.md

+3-2
Original file line numberDiff line numberDiff line change
@@ -253,8 +253,9 @@ At the top level, `variance = 1` and `scrutIsWidenedAbstract = false`.
253253
* If `P` is a `TypeWithoutCapture`:
254254
* Do nothing (always succeed).
255255
* If `P` is a `WildcardCapture` `ti = _`:
256-
* If `X` is of the form `_ >: L <: H`, instantiate `ti := H` (anything between `L` and `H` would work here),
257-
* Otherwise, instantiate `ti := X`.
256+
* Instantiate `ti` so that the subtype test in Step (2) above always succeeds:
257+
* If `X` is of the form `_ >: L <: H`, instantiate `ti := H` (resp. `L`, `X`) if `variance = 1` (resp. `-1`, `0`).
258+
* Otherwise, instantiate `ti := X`.
258259
* If `P` is a `TypeCapture` `ti`:
259260
* If `X` is of the form `_ >: L <: H`,
260261
* If `scrutIsWidenedAbstract` is `true`, fail as not specific.

_sips/sips/multiple-assignments.md

-7
This file was deleted.

_sips/sips/replace-non-sensical-unchecked-annotations.md

-7
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,260 @@
1+
---
2+
layout: sip
3+
permalink: /sips/:title.html
4+
stage: implementation
5+
status: waiting-for-implementation
6+
presip-thread: https://contributors.scala-lang.org/t/pre-sip-replace-non-sensical-unchecked-annotations/6342
7+
title: SIP-57 - Replace non-sensical @unchecked annotations
8+
---
9+
10+
**By: Martin Odersky and Jamie Thompson**
11+
12+
## History
13+
14+
| Date | Version |
15+
|---------------|--------------------|
16+
| Dec 8th 2023 | Initial Draft |
17+
| Jan 19th 2024 | Clarification about current @unchecked behavior |
18+
19+
## Summary
20+
21+
We propose to replace the mechanism to silence warnings for "unchecked" patterns, in the cases where silencing the warning will still result in the pattern being checked at runtime.
22+
23+
Currently, a user can silence warnings that a scrutinee may not be matched by a pattern, by annotating the scrutinee with the `@unchecked` annotation. This SIP proposes to use a new annotation `@RuntimeCheck` to replace `@unchecked` for this purpose. For convenience, an extension method will be added to `Predef` that marks the receiver with the annotation (used as follows: `foo.runtimeCheck`). Functionally it behaves the same as the old annotation, but improves readability at the callsite.
24+
25+
## Motivation
26+
27+
As described in [Scala 3 Reference: Pattern Bindings](https://docs.scala-lang.org/scala3/reference/changed-features/pattern-bindings.html), under `-source:future` it is an error for a pattern definition to be refutable. For instance, consider:
28+
```scala
29+
def xs: List[Any] = ???
30+
val y :: ys = xs
31+
```
32+
33+
This compiled without warning in 3.0, became a warning in 3.2, and we would like to make it an error by default in a future 3.x version.
34+
As an escape hatch we recommend to use `@unchecked`:
35+
```
36+
-- Warning: ../../new/test.scala:6:16 ------------------------------------------
37+
6 | val y :: ys = xs
38+
| ^^
39+
|pattern's type ::[Any] is more specialized than the right hand side expression's type List[Any]
40+
|
41+
|If the narrowing is intentional, this can be communicated by adding `: @unchecked` after the expression,
42+
|which may result in a MatchError at runtime.
43+
```
44+
Similarly for non-exhaustive `match` expressions, where we also recommend to put `@unchecked` on the scrutinee.
45+
46+
But `@unchecked` has several problems. First, it is ergonomically bad. For instance to fix the exhaustivity warning in
47+
```scala
48+
xs match
49+
case y :: ys => ...
50+
```
51+
we'd have to write
52+
```
53+
(xs: @unchecked) match
54+
case y :: ys => ...
55+
```
56+
Having to wrap the `@unchecked` in parentheses requires editing in two places, and arguably harms readability: both due to the churn in extra symbols, and because in this use case the `@unchecked` annotation poorly communicates intent.
57+
58+
Nominally, the purpose of the annotation is to silence warnings (_from the [API docs](https://www.scala-lang.org/api/3.3.1/scala/unchecked.html#)_):
59+
> An annotation to designate that the annotated entity should not be considered for additional compiler checks.
60+
61+
_The exact meaning of this description is open to interpretation, leading to differences between Scala 2.13 and Scala 3.x. See the [misinterpretation](#misinterpretation-of-unchecked) annex for more._
62+
63+
64+
In the following code however, the word `unchecked` is a misnomer, so could be confused for another meaning by an inexperienced user:
65+
66+
```scala
67+
def xs: List[Any] = ???
68+
val y :: ys = xs: @unchecked
69+
```
70+
After all, the pattern `y :: ys` _is_ checked, but it is done at runtime (by looking at the runtime class), rather than statically.
71+
72+
As a direct contradiction, in the following usage of `unchecked`, the meaning is the opposite:
73+
```scala
74+
xs match
75+
case ints: List[Int @unchecked] =>
76+
```
77+
Here, `@unchecked` means that the `Int` parameter will _not_ be checked at runtime: The compiler instead trusts the user that `ints` is a `List[Int]`. This could lead to a `ClassCastException` in an unrelated piece of code that uses `ints`, possibly without leaving a clear breadcrumb trail of where the faulty cast originally occurred.
78+
79+
## Proposed solution
80+
81+
### High-level overview
82+
83+
This SIP proposes to fix the ergnomics and readability of `@unchecked` in the usage where it means "checked at runtime", by instead adding a new annotation `scala.internal.RuntimeCheck`.
84+
85+
```scala
86+
package scala.annotation.internal
87+
88+
final class RuntimeCheck extends Annotation
89+
```
90+
91+
In all usages where the compiler looks for `@unchecked` for this purpose, we instead change to look for `@RuntimeCheck`.
92+
93+
By placing the annotation in the `internal` package, we communicate that the user is not meant to directly use the annotation.
94+
95+
Instead, for convenience, we provide an extension method `Predef.runtimeCheck`, which can be applied to any expression.
96+
97+
The new usage to assert that a pattern is checked at runtime then becomes as follows:
98+
```scala
99+
def xs: List[Any] = ???
100+
val y :: ys = xs.runtimeCheck
101+
```
102+
103+
We also make `runtimeCheck` a transparent inline method. This ensures that the elaboration of the method defines its semantics. (i.e. `runtimeCheck` is not meaningful because it is immediately inlined at type-checking).
104+
105+
### Specification
106+
107+
The addition of a new `scala.Predef` method:
108+
109+
```scala
110+
package scala
111+
112+
import scala.annotation.internal.RuntimeCheck
113+
114+
object Predef:
115+
extension [T](x: T)
116+
transparent inline def runtimeCheck: x.type =
117+
x: @RuntimeCheck
118+
```
119+
120+
### Compatibility
121+
122+
This change carries the usual backward binary and TASTy compatibility concerns as any other standard library addition to the Scala 3 only library.
123+
124+
Considering backwards source compatibility, the following situation will change:
125+
126+
```scala
127+
// source A.scala
128+
package example
129+
130+
extension (predef: scala.Predef.type)
131+
transparent inline def runtimeCheck[T](x: T): x.type =
132+
println("fake runtimeCheck")
133+
x
134+
```
135+
```scala
136+
// source B.scala
137+
package example
138+
139+
@main def Test =
140+
val xs = List[Any](1,2,3)
141+
val y :: ys = Predef.runtimeCheck(xs)
142+
assert(ys == List(2, 3))
143+
```
144+
145+
Previously this code would print `fake runtimeCheck`, however with the proposed change then recompiling this code will _succeed_ and no longer will print.
146+
147+
Potentially we could mitigate this if necessary with a migration warning when the new method is resolved (`@experimental` annotation would be a start)
148+
149+
150+
In general however, the new `runtimeCheck` method will not change any previously linking method without causing an ambiguity compilation error.
151+
152+
### Other concerns
153+
154+
In 3.3 we already require the user to put `@unchecked` to avoid warnings, there is likely a significant amount of existing code that will need to migrate to the new mechanism. (We can leverage already exisiting mechanisms help migrate code automatically).
155+
156+
### Open questions
157+
158+
1) A large question was should the method or annotation carry semantic weight in the language. In this proposal we weigh towards the annotation being the significant element.
159+
The new method elaborates to an annotated expression before the associated pattern exhaustivity checks occur.
160+
2) Another point, where should the helper method go? In Predef it requires no import, but another possible location was the `compiletime` package. Requiring the extra import could discourage usage without consideration - however if the method remains in `Predef` the name itself (and documentation) should signal danger, like with `asInstanceOf`.
161+
162+
3) Should the `RuntimeCheck` annotation be in the `scala.annotation.internal` package?
163+
164+
### Misinterpretation of unchecked
165+
166+
We would further like to highlight that the `unchecked` annotation is unspecified except for its imprecise API documentation. This leads to a crucial difference in its behavior between Scala 2.13 and the latest Scala 3.3.1 release.
167+
168+
#### Scala 3 semantics
169+
170+
Say you have the following:
171+
```scala
172+
val xs = List(1: Any)
173+
```
174+
175+
The following expression in Scala 3.3.1 yields two warnings:
176+
```scala
177+
xs match {
178+
case is: ::[Int] => is.head
179+
}
180+
```
181+
182+
```scala
183+
2 warnings found
184+
-- [E029] Pattern Match Exhaustivity Warning: ----------------------------------
185+
1 |xs match {
186+
|^^
187+
|match may not be exhaustive.
188+
|
189+
|It would fail on pattern case: List(_, _*), Nil
190+
|
191+
| longer explanation available when compiling with `-explain`
192+
val res0: Int = 1
193+
-- Unchecked Warning: ----------------------------------------------------------
194+
2 | case is: ::[Int] => is.head
195+
| ^
196+
|the type test for ::[Int] cannot be checked at runtime because its type arguments can't be determined from List[Any]
197+
```
198+
199+
using `@unchecked` on `xs` has the effect of silencing any warnings that depend on checking `xs`, so no warnings will be emitted for the following change:
200+
201+
```scala
202+
(xs: @unchecked) match {
203+
case is: ::[Int] => is.head
204+
}
205+
```
206+
207+
#### Scala 2.13 semantics
208+
209+
However, in Scala 2.13, this will only silence the `match may not be exhaustive` warning, and the user will still see the `type test for ::[Int] cannot be checked at runtime` warning:
210+
211+
```scala
212+
scala> (xs: @unchecked) match {
213+
| case is: ::[Int] => is.head
214+
| } ^
215+
On line 2: warning: non-variable type argument Int in type pattern scala.collection.immutable.::[Int] (the underlying of ::[Int]) is unchecked since it is eliminated by erasure
216+
val res2: Int = 1
217+
```
218+
219+
#### Aligning to Scala 2.13 semantics with runtimeCheck
220+
221+
with `xs.runtimeCheck` we should still produce an unchecked warning for `case is: ::[Int] =>`
222+
```scala
223+
scala> xs.runtimeChecked match {
224+
| case is: ::[Int] => is.head
225+
| }
226+
1 warning found
227+
-- Unchecked Warning: ----------------------------------------------------------
228+
2 | case is: ::[Int] => is.head
229+
| ^
230+
|the type test for ::[Int] cannot be checked at runtime because its type arguments can't be determined from List[Any]
231+
val res13: Int = 1
232+
```
233+
This is because `xs.runtimeChecked` means trust the user as long as the pattern can be checked at runtime.
234+
235+
To fully avoid warnings, the `@unchecked` will be put on the type argument:
236+
```scala
237+
scala> xs.runtimeChecked match {
238+
| case is: ::[Int @unchecked] => is.head
239+
| }
240+
val res14: Int = 1
241+
```
242+
This has a small extra migration cost because if the scrutinee changes from `(xs: @unchecked)` to `xs.runtimeCheck` now some individual cases might need to add `@unchecked` on type arguments to avoid creating new warnings - however this cost is offset by perhaps revealing unsafe patterns previously unaccounted for.
243+
244+
Once again `@nowarn` can be used to fully restore any old behavior
245+
246+
## Alternatives
247+
248+
1) make `runtimeCheck` a method on `Any` that returns the receiver (not inline). The compiler would check for presence of a call to this method when deciding to perform static checking of pattern exhaustivity. This idea was criticised for being brittle with respect to refactoring, or automatic code transformations via macro.
249+
250+
2) `runtimeCheck` should elaborate to code that matches the expected type, e.g. to heal `t: Any` to `Int` when the expected type is `Int`. The problem is that this is not useful for patterns that can not be runtime checked by type alone. Also, it implies a greater change to the spec, because now `runtimeCheck` would have to be specially treated.
251+
252+
## Related work
253+
254+
- [Pre SIP thread](https://contributors.scala-lang.org/t/pre-sip-replace-non-sensical-unchecked-annotations/6342)
255+
- [Scala 3 Reference: Pattern Bindings](https://docs.scala-lang.org/scala3/reference/changed-features/pattern-bindings.html),
256+
- None of OCaml, Rust, Swift, or Java offer explicit escape hatches for non-exhaustive pattern matches (Haskell does not even warn by default). Instead the user must add a default case, (making it exhaustive) or use the equivalent of `@nowarn` when they exist.
257+
258+
## FAQ
259+
260+
N/A so far.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
---
2+
title: SIP-61 - Unroll Default Arguments for Binary Compatibility
3+
status: under-review
4+
pull-request-number: 78
5+
stage: design
6+
7+
---

0 commit comments

Comments
 (0)