|
| 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. |
0 commit comments