blob: 94753bb8a82b646105c387ce006176c3dff1d093 [file] [log] [blame]
/**
* Defines a `Dsymbol` for `version = identifier` and `debug = identifier` statements.
*
* Specification: $(LINK2 https://dlang.org/spec/version.html#version-specification, Version Specification),
* $(LINK2 https://dlang.org/spec/version.html#debug_specification, Debug Specification).
*
* Copyright: Copyright (C) 1999-2022 by The D Language Foundation, All Rights Reserved
* Authors: $(LINK2 https://www.digitalmars.com, Walter Bright)
* License: $(LINK2 https://www.boost.org/LICENSE_1_0.txt, Boost License 1.0)
* Source: $(LINK2 https://github.com/dlang/dmd/blob/master/src/dmd/dversion.d, _dversion.d)
* Documentation: https://dlang.org/phobos/dmd_dversion.html
* Coverage: https://codecov.io/gh/dlang/dmd/src/master/src/dmd/dversion.d
*/
module dmd.dversion;
import dmd.arraytypes;
import dmd.cond;
import dmd.dmodule;
import dmd.dscope;
import dmd.dsymbol;
import dmd.dsymbolsem;
import dmd.globals;
import dmd.identifier;
import dmd.common.outbuffer;
import dmd.visitor;
/***********************************************************
* DebugSymbol's happen for statements like:
* debug = identifier;
* debug = integer;
*/
extern (C++) final class DebugSymbol : Dsymbol
{
uint level;
extern (D) this(const ref Loc loc, Identifier ident)
{
super(loc, ident);
}
extern (D) this(const ref Loc loc, uint level)
{
super(loc, null);
this.level = level;
}
override DebugSymbol syntaxCopy(Dsymbol s)
{
assert(!s);
auto ds = new DebugSymbol(loc, ident);
ds.comment = comment;
ds.level = level;
return ds;
}
override const(char)* toChars() const nothrow
{
if (ident)
return ident.toChars();
else
{
OutBuffer buf;
buf.print(level);
return buf.extractChars();
}
}
override void addMember(Scope* sc, ScopeDsymbol sds)
{
//printf("DebugSymbol::addMember('%s') %s\n", sds.toChars(), toChars());
Module m = sds.isModule();
// Do not add the member to the symbol table,
// just make sure subsequent debug declarations work.
if (ident)
{
if (!m)
{
error("declaration must be at module level");
errors = true;
}
else
{
if (findCondition(m.debugidsNot, ident))
{
error("defined after use");
errors = true;
}
if (!m.debugids)
m.debugids = new Identifiers();
m.debugids.push(ident);
}
}
else
{
if (!m)
{
error("level declaration must be at module level");
errors = true;
}
else
m.debuglevel = level;
}
}
override const(char)* kind() const nothrow
{
return "debug";
}
override inout(DebugSymbol) isDebugSymbol() inout
{
return this;
}
override void accept(Visitor v)
{
v.visit(this);
}
}
/***********************************************************
* VersionSymbol's happen for statements like:
* version = identifier;
* version = integer;
*/
extern (C++) final class VersionSymbol : Dsymbol
{
uint level;
extern (D) this(const ref Loc loc, Identifier ident)
{
super(loc, ident);
}
extern (D) this(const ref Loc loc, uint level)
{
super(loc, null);
this.level = level;
}
override VersionSymbol syntaxCopy(Dsymbol s)
{
assert(!s);
auto ds = ident ? new VersionSymbol(loc, ident)
: new VersionSymbol(loc, level);
ds.comment = comment;
return ds;
}
override const(char)* toChars() const nothrow
{
if (ident)
return ident.toChars();
else
{
OutBuffer buf;
buf.print(level);
return buf.extractChars();
}
}
override void addMember(Scope* sc, ScopeDsymbol sds)
{
//printf("VersionSymbol::addMember('%s') %s\n", sds.toChars(), toChars());
Module m = sds.isModule();
// Do not add the member to the symbol table,
// just make sure subsequent debug declarations work.
if (ident)
{
VersionCondition.checkReserved(loc, ident.toString());
if (!m)
{
error("declaration must be at module level");
errors = true;
}
else
{
if (findCondition(m.versionidsNot, ident))
{
error("defined after use");
errors = true;
}
if (!m.versionids)
m.versionids = new Identifiers();
m.versionids.push(ident);
}
}
else
{
if (!m)
{
error("level declaration must be at module level");
errors = true;
}
else
m.versionlevel = level;
}
}
override const(char)* kind() const nothrow
{
return "version";
}
override inout(VersionSymbol) isVersionSymbol() inout
{
return this;
}
override void accept(Visitor v)
{
v.visit(this);
}
}