[challenge] Limitation in D's metaprogramming

Robert Jacques sandford at jhu.edu
Mon Oct 18 21:37:33 PDT 2010


On Mon, 18 Oct 2010 20:07:16 -0400, Nick Sabalausky <a at a.a> wrote:
> I don't know if others have noticed this before, but I think I've found a
> notable limitation in D's metaprogramming potential: There doesn't  
> appear to
> be a way to have mutable global state at compile-time.
>
> Challange:
>
> Create two..."things"...they can be functions, templates, variables,
> mix-n-match, whatever. One of them increments a counter, and the other  
> can
> be used to retreive the value. But both of these must operate at
> compile-time, and they must both be callable (directly or indirectly,
> doesn't matter) from within the context of any module that imports them.
>
> This is an example, except it operates at run-time, not compile-time:
>
> -----------------------------------
> // a.d
> module a;
> int value=0;
> void inc()
> {
>     value++;
> }
> int get()
> {
>     return value;
> }
>
> void incFromA()
> {
>     inc();
> }
>
> //b.d
> module b;
> import a;
> void incFromB()
> {
>     inc();
> }
>
> //main.d
> import a;
> import b;
> import std.stdio;
> void main()
> {
>     inc();
>     incFromA();
>     incFromB();
>     writeln(get());
> }
> -----------------------------------
>
> The goal of this challenge is to define a global-level manifest constant
> enum that holds a value that has been incremented from multiple modules:
>
> enum GOAL = ????;
>
> It can, of course, then be displayed via:
> pragma(msg, std.conv.to!string(GOAL));
>
> At this point, I'm not concerned about order-of-execution issues  
> resulting
> in unexpected or unpredictable values. As long as a value can be  
> incremented
> at compile-time from multiple modules and used to initialize an enum
> manifest constant, that satisfies this challenge.
>

This isn't exactly what you're looking for, but you can abuse conditional  
compilation + D's symbol table to create a scoped counter:

string nthLabel(int n) {
     return "__Global_"~ to!string(n);
}
string incGlobal() {
     return `mixin("alias int "~nthLabel( getGlobal!(__FILE__, __LINE__)  
)~";");`;
}
template getGlobal(string file = __FILE__, int line = __LINE__, int N = 0)  
{
     static if( !__traits(compiles, mixin(nthLabel(N)) ) )
         enum getGlobal = N;
     else
         enum getGlobal = getGlobal!(file,line, N+1 );
}

enum Foo    = getGlobal!(__FILE__, __LINE__) ;
mixin( incGlobal() );
enum Bar    = getGlobal!(__FILE__, __LINE__);
mixin( incGlobal() );
enum FooBar = getGlobal!(__FILE__, __LINE__);


void main(string[] args) {
     //      0        1        2
     writeln(Foo,'\t',Bar,'\t',FooBar);
     return;
}


More information about the Digitalmars-d mailing list