diff options
author | Andreas Rumpf <rumpf_a@web.de> | 2022-10-07 22:26:53 +0200 |
---|---|---|
committer | GitHub <noreply@github.com> | 2022-10-07 22:26:53 +0200 |
commit | e83f27e6a0f52f167e8eb91cd8f60be62d6725c6 (patch) | |
tree | 5e23e0c54abe38038405992deea1c234030d49b1 | |
parent | a132f5502acbd53781802579d89d6ca5168e74cd (diff) | |
download | Nim-e83f27e6a0f52f167e8eb91cd8f60be62d6725c6.tar.gz |
out parameters: enforce that 'out' is only used as a parameter (#20510)
* out parameters: enforce that 'out' is only used as a parameter * make tables.nim use 'out' parameters * better backwards compat
-rw-r--r-- | compiler/typeallowed.nim | 2 | ||||
-rw-r--r-- | lib/core/macros.nim | 5 | ||||
-rw-r--r-- | lib/pure/collections/hashcommon.nim | 3 | ||||
-rw-r--r-- | lib/pure/collections/tableimpl.nim | 2 | ||||
-rw-r--r-- | lib/std/outparams.nim | 38 |
5 files changed, 47 insertions, 3 deletions
diff --git a/compiler/typeallowed.nim b/compiler/typeallowed.nim index 57dd039ad..37b548865 100644 --- a/compiler/typeallowed.nim +++ b/compiler/typeallowed.nim @@ -65,6 +65,8 @@ proc typeAllowedAux(marker: var IntSet, typ: PType, kind: TSymKind, elif t.kind == tyLent and ((kind != skResult and views notin c.features) or kind == skParam): # lent can't be used as parameters. result = t + elif isOutParam(t) and kind != skParam: + result = t else: var t2 = skipTypes(t[0], abstractInst-{tyTypeDesc, tySink}) case t2.kind diff --git a/lib/core/macros.nim b/lib/core/macros.nim index cc2d5041e..595d4bab5 100644 --- a/lib/core/macros.nim +++ b/lib/core/macros.nim @@ -75,7 +75,7 @@ type nnkTupleTy, nnkTupleClassTy, nnkTypeClassTy, nnkStaticTy, nnkRecList, nnkRecCase, nnkRecWhen, nnkRefTy, nnkPtrTy, nnkVarTy, - nnkConstTy, nnkMutableTy, + nnkConstTy, nnkOutTy, nnkDistinctTy, nnkProcTy, nnkIteratorTy, # iterator type @@ -125,6 +125,9 @@ type TNimSymKinds* {.deprecated.} = set[NimSymKind] +const + nnkMutableTy* {.deprecated.} = nnkOutTy + type NimIdent* {.deprecated.} = object of RootObj ## Represents a Nim identifier in the AST. **Note**: This is only diff --git a/lib/pure/collections/hashcommon.nim b/lib/pure/collections/hashcommon.nim index deff8fa21..02312e2dc 100644 --- a/lib/pure/collections/hashcommon.nim +++ b/lib/pure/collections/hashcommon.nim @@ -13,6 +13,7 @@ when defined(nimPreviewSlimSystem): import std/assertions +import std / outparams const growthFactor = 2 @@ -78,5 +79,5 @@ template rawGetImpl() {.dirty.} = genHashImpl(key, hc) rawGetKnownHCImpl() -proc rawGet[X, A](t: X, key: A, hc: var Hash): int {.inline.} = +proc rawGet[X, A](t: X, key: A, hc: var Hash): int {.inline, outParamsAt: [3].} = rawGetImpl() diff --git a/lib/pure/collections/tableimpl.nim b/lib/pure/collections/tableimpl.nim index 27c339177..81079a3d1 100644 --- a/lib/pure/collections/tableimpl.nim +++ b/lib/pure/collections/tableimpl.nim @@ -23,7 +23,7 @@ template rawInsertImpl() {.dirty.} = data[h].val = val data[h].hcode = hc -proc rawGetDeep[X, A](t: X, key: A, hc: var Hash): int {.inline.} = +proc rawGetDeep[X, A](t: X, key: A, hc: var Hash): int {.inline, outParamsAt: [3].} = rawGetDeepImpl() proc rawInsert[X, A, B](t: var X, data: var KeyValuePairSeq[A, B], diff --git a/lib/std/outparams.nim b/lib/std/outparams.nim new file mode 100644 index 000000000..8a0e5ae67 --- /dev/null +++ b/lib/std/outparams.nim @@ -0,0 +1,38 @@ +# +# +# Nim's Runtime Library +# (c) Copyright 2022 Nim contributors +# +# See the file "copying.txt", included in this +# distribution, for details about the copyright. +# + +## `outParamsAt` macro for easy writing code that works with both 2.0 and 1.x. + +import macros + +macro outParamsAt*(positions: static openArray[int]; n: untyped): untyped = + ## Use this macro to annotate `out` parameters in a portable way. + runnableExamples: + proc p(x: var int) {.outParamsAt: [1].} = + discard "x is really an 'out int' if the Nim compiler supports 'out' parameters" + + result = n + when defined(nimHasOutParams): + var p = n.params + for po in positions: + p[po][^2].expectKind nnkVarTy + p[po][^2] = newTree(nnkOutTy, p[po][^2][0]) + +when isMainModule: + {.experimental: "strictDefs".} + + proc main(x: var int) {.outParamsAt: [1].} = + x = 3 + + proc us = + var x: int + main x + echo x + + us() |