diff options
author | metagn <metagngn@gmail.com> | 2024-09-11 10:05:39 +0300 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-09-11 09:05:39 +0200 |
commit | 771369237c579afb93935c8bef5e3c79155ddfd6 (patch) | |
tree | d64c54eb1cfcbab4ab9048f0cc6c794ae963f53d /tests | |
parent | baec1955b5453ec71fc12355142dd9a813fa02fb (diff) | |
download | Nim-771369237c579afb93935c8bef5e3c79155ddfd6.tar.gz |
implement template default values using other params (#24073)
fixes #23506 #24065 broke compilation of template parameter default values that depended on other template parameters. But this was never implemented anyway, actually attempting to use those default values breaks the compiler as in #23506. So these are now implemented as well as fixing the regression. First, if a default value expression uses any unresolved arguments (generic or normal template parameters) in a template header, we leave it untyped, instead of applying the generic typechecking (fixing the regression). Then, just before the body of the template is about to be explored, the default value expressions are handled in the same manner as the body as well. This captures symbols including the parameters, so the expression is checked again if it contains a parameter symbol, and marked with `nfDefaultRefsParam` if it does (as an optimization to not check it later). Then when the template is being evaluated, when substituting a parameter, if we try to substitute with a node marked `nfDefaultRefsParam`, we also evaluate it as we would the template body instead of passing it as just a copy (the reason why it never worked before). This way we save time if a default value doesn't refer to another parameter and could just be copied regardless.
Diffstat (limited to 'tests')
-rw-r--r-- | tests/template/tdefaultparam.nim | 15 |
1 files changed, 15 insertions, 0 deletions
diff --git a/tests/template/tdefaultparam.nim b/tests/template/tdefaultparam.nim new file mode 100644 index 000000000..ecbb0e145 --- /dev/null +++ b/tests/template/tdefaultparam.nim @@ -0,0 +1,15 @@ +block: + template foo(a: untyped, b: untyped = a(0)): untyped = + let x = a(0) + let y = b + (x, y) + proc bar(x: int): int = x + 1 + doAssert foo(bar, b = bar(0)) == (1, 1) + doAssert foo(bar) == (1, 1) + +block: # issue #23506 + var a: string + template foo(x: int; y = x) = + a = $($x, $y) + foo(1) + doAssert a == "(\"1\", \"1\")" |