Skip to content

Commit 751bc50

Browse files
Support more types in simple for-loops
1 parent 898c6c7 commit 751bc50

22 files changed

+175
-14
lines changed

docs/release-notes/.FSharp.Compiler.Service/9.0.300.md

+1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
* Add support for C# `Experimental` attribute. ([PR #18253](https://github.com/dotnet/fsharp/pull/18253))
1818
* Nullness warnings are issued for signature<>implementation conformance ([PR #18186](https://github.com/dotnet/fsharp/pull/18186))
1919
* Symbols: Add FSharpAssembly.IsFSharp ([PR #18290](https://github.com/dotnet/fsharp/pull/18290))
20+
* Support more types in simple for-loops. ([Language suggestion #876](https://github.com/fsharp/fslang-suggestions/issues/876), [PR #18301](https://github.com/dotnet/fsharp/pull/18301))
2021

2122
### Changed
2223

docs/release-notes/.Language/preview.md

+1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
* Deprecate places where `seq` can be omitted. ([Language suggestion #1033](https://github.com/fsharp/fslang-suggestions/issues/1033), [PR #17772](https://github.com/dotnet/fsharp/pull/17772))
55
* Added type conversions cache, only enabled for compiler runs ([PR#17668](https://github.com/dotnet/fsharp/pull/17668))
66
* Support ValueOption + Struct attribute as optional parameter for methods ([Language suggestion #1136](https://github.com/fsharp/fslang-suggestions/issues/1136), [PR #18098](https://github.com/dotnet/fsharp/pull/18098))
7+
* Support more types in simple for-loops. ([Language suggestion #876](https://github.com/fsharp/fslang-suggestions/issues/876), [PR #18301](https://github.com/dotnet/fsharp/pull/18301))
78

89
### Fixed
910

src/Compiler/Checking/Expressions/CheckExpressions.fs

+54-14
Original file line numberDiff line numberDiff line change
@@ -6271,24 +6271,64 @@ and TcExprIntegerForLoop (cenv: cenv) overallTy env tpenv (spFor, spTo, id, star
62716271
let g = cenv.g
62726272
UnifyTypes cenv env m overallTy.Commit g.unit_ty
62736273

6274-
let startExpr, tpenv =
6275-
let env = { env with eIsControlFlow = false }
6276-
TcExpr cenv (MustEqual g.int_ty) env tpenv start
6274+
let tryTcStartAndFinishAsInt32 tpenv start finish =
6275+
let tcDefaultInt32 tpenv (synExpr: SynExpr) =
6276+
let addCxTyparDefaultsToInt32 ty =
6277+
tryDestTyparTy g ty
6278+
|> ValueOption.iter (fun typar ->
6279+
AddCxTyparDefaultsTo env.DisplayEnv cenv.css synExpr.Range env.eContextInfo typar 1 g.int_ty)
62776280

6278-
let finishExpr, tpenv =
6279-
let env = { env with eIsControlFlow = false }
6280-
TcExpr cenv (MustEqual g.int_ty) env tpenv finish
6281+
let exprTy = NewInferenceType g
6282+
addCxTyparDefaultsToInt32 exprTy
6283+
let env = { env with eIsControlFlow = false }
6284+
let expr, tpenv = TcExpr cenv (MustEqual exprTy) env tpenv synExpr
6285+
expr, exprTy, tpenv
62816286

6282-
let idv, _ = mkLocal id.idRange id.idText g.int_ty
6283-
let envinner = AddLocalVal g cenv.tcSink m idv env
6284-
let envinner = { envinner with eIsControlFlow = true }
6287+
let startExpr, startTy, tpenv = tcDefaultInt32 tpenv start
6288+
let finishExpr, finishTy, tpenv = tcDefaultInt32 tpenv finish
6289+
6290+
if typeEquiv g startTy g.int_ty && typeEquiv g finishTy g.int_ty then
6291+
Some (tpenv, startExpr, finishExpr)
6292+
else
6293+
None
6294+
6295+
// First try to typecheck the start and finish expressions as int32
6296+
// for backwards compatibility. Otherwise, treat the for-loop
6297+
// as though it were a for-each loop over a range expression.
6298+
match tryTcStartAndFinishAsInt32 tpenv start finish with
6299+
| Some (tpenv, startExpr, finishExpr) ->
6300+
let idv, _ = mkLocal id.idRange id.idText g.int_ty
6301+
let envinner = AddLocalVal g cenv.tcSink m idv env
6302+
let envinner = { envinner with eIsControlFlow = true }
6303+
6304+
// notify name resolution sink about loop variable
6305+
let item = Item.Value(mkLocalValRef idv)
6306+
CallNameResolutionSink cenv.tcSink (idv.Range, env.NameEnv, item, emptyTyparInst, ItemOccurrence.Binding, env.AccessRights)
62856307

6286-
// notify name resolution sink about loop variable
6287-
let item = Item.Value(mkLocalValRef idv)
6288-
CallNameResolutionSink cenv.tcSink (idv.Range, env.NameEnv, item, emptyTyparInst, ItemOccurrence.Binding, env.AccessRights)
6308+
let bodyExpr, tpenv = TcStmt cenv envinner tpenv body
6309+
mkFastForLoop g (spFor, spTo, m, idv, startExpr, dir, finishExpr, bodyExpr), tpenv
62896310

6290-
let bodyExpr, tpenv = TcStmt cenv envinner tpenv body
6291-
mkFastForLoop g (spFor, spTo, m, idv, startExpr, dir, finishExpr, bodyExpr), tpenv
6311+
| None ->
6312+
// TODO: Figure this out.
6313+
//checkLanguageFeatureAndRecover g.langVersion LanguageFeature.MoreTypesInSimpleForLoops m
6314+
let pat = SynPat.Named (SynIdent (id, None), false, None, id.idRange)
6315+
6316+
let rangeExpr =
6317+
let mTo = match spTo with DebugPointAtInOrTo.Yes m -> m | DebugPointAtInOrTo.No -> Range.range0
6318+
6319+
if dir then
6320+
// for x = start to finish do …
6321+
// → for x in start..finish do …
6322+
mkSynInfix mTo start ".." finish
6323+
else
6324+
// for x = start downto finish do …
6325+
// → for x in start..-1..finish do …
6326+
let minus = mkSynOperator mTo "~-"
6327+
let one = mkSynLidGet mTo ["Microsoft"; "FSharp"; "Core"; "LanguagePrimitives"] "GenericOne"
6328+
let step = mkSynApp1 minus one mTo
6329+
mkSynTrifix (mTo.MakeSynthetic()) ".. .." start step finish
6330+
6331+
TcForEachExpr cenv overallTy env tpenv (false, true, pat, rangeExpr, body, m, spFor, spTo, m)
62926332

62936333
and TcExprTryWith (cenv: cenv) overallTy env tpenv (synBodyExpr, synWithClauses, mWithToLast, mTryToLast, spTry, spWith) =
62946334
let g = cenv.g

src/Compiler/FSComp.txt

+1
Original file line numberDiff line numberDiff line change
@@ -1794,3 +1794,4 @@ featureDontWarnOnUppercaseIdentifiersInBindingPatterns,"Don't warn on uppercase
17941794
3874,tcExpectedTypeParamMarkedWithUnitOfMeasureAttribute,"Expected unit-of-measure type parameter must be marked with the [<Measure>] attribute."
17951795
featureDeprecatePlacesWhereSeqCanBeOmitted,"Deprecate places where 'seq' can be omitted"
17961796
featureSupportValueOptionsAsOptionalParameters,"Support ValueOption as valid type for optional member parameters"
1797+
featureMoreTypesInSimpleForLoops,"Support more types in simple for-loops"

src/Compiler/Facilities/LanguageFeatures.fs

+3
Original file line numberDiff line numberDiff line change
@@ -98,6 +98,7 @@ type LanguageFeature =
9898
| UseTypeSubsumptionCache
9999
| DeprecatePlacesWhereSeqCanBeOmitted
100100
| SupportValueOptionsAsOptionalParameters
101+
| MoreTypesInSimpleForLoops
101102

102103
/// LanguageVersion management
103104
type LanguageVersion(versionText) =
@@ -227,6 +228,7 @@ type LanguageVersion(versionText) =
227228
LanguageFeature.DontWarnOnUppercaseIdentifiersInBindingPatterns, previewVersion
228229
LanguageFeature.DeprecatePlacesWhereSeqCanBeOmitted, previewVersion
229230
LanguageFeature.SupportValueOptionsAsOptionalParameters, previewVersion
231+
LanguageFeature.MoreTypesInSimpleForLoops, previewVersion
230232
]
231233

232234
static let defaultLanguageVersion = LanguageVersion("default")
@@ -388,6 +390,7 @@ type LanguageVersion(versionText) =
388390
| LanguageFeature.UseTypeSubsumptionCache -> FSComp.SR.featureUseTypeSubsumptionCache ()
389391
| LanguageFeature.DeprecatePlacesWhereSeqCanBeOmitted -> FSComp.SR.featureDeprecatePlacesWhereSeqCanBeOmitted ()
390392
| LanguageFeature.SupportValueOptionsAsOptionalParameters -> FSComp.SR.featureSupportValueOptionsAsOptionalParameters ()
393+
| LanguageFeature.MoreTypesInSimpleForLoops -> FSComp.SR.featureMoreTypesInSimpleForLoops ()
391394

392395
/// Get a version string associated with the given feature.
393396
static member GetFeatureVersionString feature =

src/Compiler/Facilities/LanguageFeatures.fsi

+1
Original file line numberDiff line numberDiff line change
@@ -89,6 +89,7 @@ type LanguageFeature =
8989
| UseTypeSubsumptionCache
9090
| DeprecatePlacesWhereSeqCanBeOmitted
9191
| SupportValueOptionsAsOptionalParameters
92+
| MoreTypesInSimpleForLoops
9293

9394
/// LanguageVersion management
9495
type LanguageVersion =

src/Compiler/TypedTree/TcGlobals.fs

+4
Original file line numberDiff line numberDiff line change
@@ -731,6 +731,8 @@ type TcGlobals(
731731
let v_generic_equality_withc_outer_info = makeIntrinsicValRef(fslib_MFLanguagePrimitives_nleref, "GenericEqualityWithComparer" , None , None , [vara], mk_equality_withc_sig varaTy)
732732
let v_generic_hash_withc_outer_info = makeIntrinsicValRef(fslib_MFLanguagePrimitives_nleref, "GenericHashWithComparer" , None , None , [vara], mk_hash_withc_sig varaTy)
733733

734+
let v_generic_one_info = makeIntrinsicValRef(fslib_MFLanguagePrimitives_nleref, "GenericOne" , None , None , [], ([[v_unit_ty]], varaTy))
735+
734736
let v_generic_equality_er_inner_info = makeIntrinsicValRef(fslib_MFHashCompare_nleref, "GenericEqualityERIntrinsic" , None , None , [vara], mk_rel_sig varaTy)
735737
let v_generic_equality_per_inner_info = makeIntrinsicValRef(fslib_MFHashCompare_nleref, "GenericEqualityIntrinsic" , None , None , [vara], mk_rel_sig varaTy)
736738
let v_generic_equality_withc_inner_info = makeIntrinsicValRef(fslib_MFHashCompare_nleref, "GenericEqualityWithComparerIntrinsic" , None , None , [vara], mk_equality_withc_sig varaTy)
@@ -1591,6 +1593,8 @@ type TcGlobals(
15911593
member val generic_hash_inner_vref = ValRefForIntrinsic v_generic_hash_inner_info
15921594
member val generic_hash_withc_inner_vref = ValRefForIntrinsic v_generic_hash_withc_inner_info
15931595

1596+
member val generic_one_vref = ValRefForIntrinsic v_generic_one_info
1597+
15941598
member val reference_equality_inner_vref = ValRefForIntrinsic v_reference_equality_inner_info
15951599

15961600
member val piperight_vref = ValRefForIntrinsic v_piperight_info

src/Compiler/TypedTree/TcGlobals.fsi

+2
Original file line numberDiff line numberDiff line change
@@ -724,6 +724,8 @@ type internal TcGlobals =
724724

725725
member generic_hash_withc_outer_info: IntrinsicValRef
726726

727+
member generic_one_vref: FSharp.Compiler.TypedTree.ValRef
728+
727729
member generic_hash_withc_tuple2_vref: FSharp.Compiler.TypedTree.ValRef
728730

729731
member generic_hash_withc_tuple3_vref: FSharp.Compiler.TypedTree.ValRef

src/Compiler/TypedTree/TypedTreeOps.fs

+43
Original file line numberDiff line numberDiff line change
@@ -7647,6 +7647,17 @@ let mkTypedOne g m ty =
76477647
elif typeEquivAux EraseMeasures g ty g.decimal_ty then Expr.Const (Const.Decimal 1m, m, ty)
76487648
else error (InternalError ($"Unrecognized numeric type '{ty}'.", m))
76497649

7650+
let mkTypedMinusOne g m ty =
7651+
if typeEquivAux EraseMeasures g ty g.int32_ty then Expr.Const (Const.Int32 -1, m, ty)
7652+
elif typeEquivAux EraseMeasures g ty g.int64_ty then Expr.Const (Const.Int64 -1L, m, ty)
7653+
elif typeEquivAux EraseMeasures g ty g.nativeint_ty then Expr.Const (Const.IntPtr -1L, m, ty)
7654+
elif typeEquivAux EraseMeasures g ty g.int16_ty then Expr.Const (Const.Int16 -1s, m, ty)
7655+
elif typeEquivAux EraseMeasures g ty g.sbyte_ty then Expr.Const (Const.SByte -1y, m, ty)
7656+
elif typeEquivAux EraseMeasures g ty g.float32_ty then Expr.Const (Const.Single -1.0f, m, ty)
7657+
elif typeEquivAux EraseMeasures g ty g.float_ty then Expr.Const (Const.Double -1.0, m, ty)
7658+
elif typeEquivAux EraseMeasures g ty g.decimal_ty then Expr.Const (Const.Decimal -1m, m, ty)
7659+
else error (InternalError ($"Unrecognized or unsigned numeric type '{ty}'.", m))
7660+
76507661
let destInt32 = function Expr.Const (Const.Int32 n, _, _) -> Some n | _ -> None
76517662

76527663
let isIDelegateEventType g ty =
@@ -10460,8 +10471,26 @@ let (|Let|_|) expr =
1046010471
| Expr.Let (TBind(v, e1, sp), e2, _, _) -> ValueSome(v, e1, sp, e2)
1046110472
| _ -> ValueNone
1046210473

10474+
/// Microsoft.FSharp.Core.LanguagePrimitives.GenericOne
10475+
let (|GenericOne|_|) g expr =
10476+
match expr with
10477+
| Expr.Val (vref, _, _) -> valRefEq g vref g.generic_one_vref
10478+
| _ -> false
10479+
10480+
/// Microsoft.FSharp.Core.Operators.(~-)
10481+
let (|UnaryMinus|_|) g expr =
10482+
match expr with
10483+
| Expr.Val (vref, _, _) -> valRefEq g vref g.unchecked_unary_minus_vref
10484+
| _ -> false
10485+
1046310486
[<return: Struct>]
1046410487
let (|RangeInt32Step|_|) g expr =
10488+
let (|GenericPlusOrMinusOne|_|) g expr =
10489+
match expr with
10490+
| Expr.App (funcExpr = UnaryMinus g; args = [Expr.App (funcExpr = GenericOne g)]) -> ValueSome -1
10491+
| Expr.App (funcExpr = GenericOne g) -> ValueSome 1
10492+
| _ -> ValueNone
10493+
1046510494
match expr with
1046610495
// detect 'n .. m'
1046710496
| Expr.App (Expr.Val (vf, _, _), _, [tyarg], [startExpr;finishExpr], _)
@@ -10471,6 +10500,9 @@ let (|RangeInt32Step|_|) g expr =
1047110500
| Expr.App (Expr.Val (vf, _, _), _, [], [startExpr; Int32Expr n; finishExpr], _)
1047210501
when valRefEq g vf g.range_int32_op_vref -> ValueSome(startExpr, n, finishExpr)
1047310502

10503+
| Expr.App (Expr.Val (vf, _, _), _, [ty1; ty2], [startExpr; GenericPlusOrMinusOne g n; finishExpr], _)
10504+
when valRefEq g vf g.range_step_op_vref && typeEquiv g ty1 g.int_ty && typeEquiv g ty2 g.int_ty -> ValueSome(startExpr, n, finishExpr)
10505+
1047410506
| _ -> ValueNone
1047510507

1047610508
[<return: Struct>]
@@ -11078,6 +11110,17 @@ let mkRangeCount g m rangeTy rangeExpr start step finish =
1107811110

1107911111
let mkOptimizedRangeLoop (g: TcGlobals) (mBody, mFor, mIn, spInWhile) (rangeTy, rangeExpr) (start, step, finish) (buildLoop: (Count -> ((Idx -> Elem -> Body) -> Loop) -> Expr)) =
1108011112
let inline mkLetBindingsIfNeeded f =
11113+
/// Replace LanguagePrimitives.GenericOne or -LanguagePrimitives.GenericOne with their constant equivalents.
11114+
/// -LanguagePrimitives.GenericOne is emitted in CheckExpressions.TcExprIntegerForLoop for `downto`
11115+
/// for types other than System.Int32.
11116+
let constifyPlusOrMinusGenericOne expr =
11117+
match expr with
11118+
| Expr.App (funcExpr = UnaryMinus g; args = [Expr.App (funcExpr = GenericOne g)]) -> Some (mkTypedMinusOne g expr.Range (tyOfExpr g expr))
11119+
| Expr.App (funcExpr = GenericOne g) -> Some (mkTypedOne g expr.Range (tyOfExpr g expr))
11120+
| _ -> None
11121+
11122+
let step = constifyPlusOrMinusGenericOne step |> Option.defaultValue step
11123+
1108111124
match start, step, finish with
1108211125
| (Expr.Const _ | Expr.Val _), (Expr.Const _ | Expr.Val _), (Expr.Const _ | Expr.Val _) ->
1108311126
f start step finish

src/Compiler/xlf/FSComp.txt.cs.xlf

+5
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/Compiler/xlf/FSComp.txt.de.xlf

+5
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/Compiler/xlf/FSComp.txt.es.xlf

+5
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/Compiler/xlf/FSComp.txt.fr.xlf

+5
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/Compiler/xlf/FSComp.txt.it.xlf

+5
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/Compiler/xlf/FSComp.txt.ja.xlf

+5
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/Compiler/xlf/FSComp.txt.ko.xlf

+5
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/Compiler/xlf/FSComp.txt.pl.xlf

+5
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/Compiler/xlf/FSComp.txt.pt-BR.xlf

+5
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/Compiler/xlf/FSComp.txt.ru.xlf

+5
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/Compiler/xlf/FSComp.txt.tr.xlf

+5
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)