digitalmars.D.learn - template instantiation problems
- berni44 (63/63) Jan 10 2020 I'm trying to understand issue 17441 [1]. I reduced it (including
- ag0aep6g (19/26) Jan 10 2020 ----
- Boris Carvajal (3/15) Jan 14 2020 Thanks for your test case.
I'm trying to understand issue 17441 [1]. I reduced it (including
Phobos) to the following (and added some message to point out the
problem):
test.d:
---
alias parentOf(alias sym) = __traits(parent, sym);
template packageName(alias T)
{
pragma(msg, "IN: ", T.stringof);
static if (__traits(compiles, parentOf!T))
enum parent = packageName!(parentOf!T);
else
enum string parent = null;
static if (T.stringof[0..8] == "package ")
enum packageName = (parent? parent ~ '.' : "") ~
T.stringof[8 .. $];
else
enum packageName = parent;
}
void main()
{
import mod0 = foo.baz;
import mod1 = foo;
pragma(msg, "mod0: ",mod0.stringof);
pragma(msg, "mod1: ",mod1.stringof);
pragma(msg, "A");
enum a = packageName!mod0;
pragma(msg, "B");
enum b = packageName!mod1;
pragma(msg, "C");
static assert(a == "foo");
static assert(b == ""); // Error: static assert: "foo"
== "" is false
}
---
foo/baz.d:
---
module foo.baz;
---
foo/package.d:
---
module foo;
---
When compiling with
$> dmd test.d
I get:
---
mod0: module baz
mod1: module foo
A
IN: module baz
IN: package foo
B
C
test.d(32): Error: static assert: "foo" == "" is false
---
This clearly shows, that packageName is only instatiated once at
top level although mod0 and mod1 are two different things (at
least their stringof produces different results) and therefore
should be instantiated twice.
Now I don't know if this is a bug in DMD or I should know
something I do not know... Can you help me?
[1] https://issues.dlang.org/show_bug.cgi?id=17441
Jan 10 2020
On 10.01.20 18:27, berni44 wrote:This clearly shows, that packageName is only instatiated once at top level although mod0 and mod1 are two different things (at least their stringof produces different results) and therefore should be instantiated twice. Now I don't know if this is a bug in DMD or I should know something I do not know... Can you help me?---- import m1 = foo.baz; import m2 = foo; alias p = __traits(parent, m1); pragma(msg, m1.stringof); /* "module baz", ok */ pragma(msg, m2.stringof); /* "module foo", ok */ pragma(msg, p.stringof); /* "package foo", ok */ enum e(alias thing) = thing.stringof; pragma(msg, e!m1); /* "module baz", ok */ pragma(msg, e!m2); /* "module foo", ok */ pragma(msg, e!p); /* "module foo", wat?? */ ---- If you switch the last two lines around, you get "package foo" twice. The compiler apparently thinks that m2 (module foo) and p (package foo) are the same thing when used as a template argument. So it reuses the previous result. There's a similar issue with values: https://issues.dlang.org/show_bug.cgi?id=14501
Jan 10 2020
On Friday, 10 January 2020 at 18:31:59 UTC, ag0aep6g wrote:---- import m1 = foo.baz; import m2 = foo; alias p = __traits(parent, m1); pragma(msg, m1.stringof); /* "module baz", ok */ pragma(msg, m2.stringof); /* "module foo", ok */ pragma(msg, p.stringof); /* "package foo", ok */ enum e(alias thing) = thing.stringof; pragma(msg, e!m1); /* "module baz", ok */ pragma(msg, e!m2); /* "module foo", ok */ pragma(msg, e!p); /* "module foo", wat?? */ ----Thanks for your test case. https://github.com/dlang/dmd/pull/10724
Jan 14 2020








Boris Carvajal <boris2.9 gmail.com>