mirror of
https://github.com/ldc-developers/ldc.git
synced 2025-05-05 01:20:51 +03:00

This reduces the size of a statically linked Phobos-based Hello World by 11 kB on Linux x86_64. Also creates a header file for gen/module.cpp, which has been renamed to "modules" such as not to conflict with the frontend header file.
223 lines
7.4 KiB
C++
223 lines
7.4 KiB
C++
|
|
/* Compiler implementation of the D programming language
|
|
* Copyright (c) 1999-2014 by Digital Mars
|
|
* All Rights Reserved
|
|
* written by Walter Bright
|
|
* http://www.digitalmars.com
|
|
* Distributed under the Boost Software License, Version 1.0.
|
|
* http://www.boost.org/LICENSE_1_0.txt
|
|
* https://github.com/D-Programming-Language/dmd/blob/master/src/module.h
|
|
*/
|
|
|
|
#ifndef DMD_MODULE_H
|
|
#define DMD_MODULE_H
|
|
|
|
#ifdef __DMC__
|
|
#pragma once
|
|
#endif /* __DMC__ */
|
|
|
|
#include "root.h"
|
|
#include "dsymbol.h"
|
|
|
|
class ClassDeclaration;
|
|
struct ModuleDeclaration;
|
|
struct Macro;
|
|
struct Escape;
|
|
class VarDeclaration;
|
|
class Library;
|
|
|
|
#if IN_LLVM
|
|
#include <cstdint>
|
|
class DValue;
|
|
namespace llvm {
|
|
class LLVMContext;
|
|
class Module;
|
|
class GlobalVariable;
|
|
class StructType;
|
|
}
|
|
#endif
|
|
|
|
|
|
enum PKG
|
|
{
|
|
PKGunknown, // not yet determined whether it's a package.d or not
|
|
PKGmodule, // already determined that's an actual package.d
|
|
PKGpackage, // already determined that's an actual package
|
|
};
|
|
|
|
class Package : public ScopeDsymbol
|
|
{
|
|
public:
|
|
PKG isPkgMod;
|
|
unsigned tag; // auto incremented tag, used to mask package tree in scopes
|
|
Module *mod; // != NULL if isPkgMod == PKGmodule
|
|
|
|
Package(Identifier *ident);
|
|
const char *kind();
|
|
|
|
static DsymbolTable *resolve(Identifiers *packages, Dsymbol **pparent, Package **ppkg);
|
|
|
|
Package *isPackage() { return this; }
|
|
|
|
bool isAncestorPackageOf(const Package * const pkg) const;
|
|
|
|
void semantic(Scope *sc) { }
|
|
Dsymbol *search(Loc loc, Identifier *ident, int flags = SearchLocalsOnly);
|
|
void accept(Visitor *v) { v->visit(this); }
|
|
|
|
Module *isPackageMod();
|
|
};
|
|
|
|
class Module : public Package
|
|
{
|
|
public:
|
|
static Module *rootModule;
|
|
static DsymbolTable *modules; // symbol table of all modules
|
|
static Modules amodules; // array of all modules
|
|
static Dsymbols deferred; // deferred Dsymbol's needing semantic() run on them
|
|
static Dsymbols deferred3; // deferred Dsymbol's needing semantic3() run on them
|
|
static unsigned dprogress; // progress resolving the deferred list
|
|
static void _init();
|
|
|
|
static AggregateDeclaration *moduleinfo;
|
|
|
|
|
|
const char *arg; // original argument name
|
|
ModuleDeclaration *md; // if !NULL, the contents of the ModuleDeclaration declaration
|
|
File *srcfile; // input source file
|
|
File *objfile; // output .obj file
|
|
File *hdrfile; // 'header' file
|
|
File *docfile; // output documentation file
|
|
unsigned errors; // if any errors in file
|
|
unsigned numlines; // number of lines in source file
|
|
int isDocFile; // if it is a documentation input file, not D source
|
|
bool isPackageFile; // if it is a package.d
|
|
int needmoduleinfo;
|
|
|
|
int selfimports; // 0: don't know, 1: does not, 2: does
|
|
bool selfImports(); // returns true if module imports itself
|
|
|
|
int rootimports; // 0: don't know, 1: does not, 2: does
|
|
bool rootImports(); // returns true if module imports root module
|
|
|
|
int insearch;
|
|
Identifier *searchCacheIdent;
|
|
Dsymbol *searchCacheSymbol; // cached value of search
|
|
int searchCacheFlags; // cached flags
|
|
|
|
// module from command line we're imported from,
|
|
// i.e. a module that will be taken all the
|
|
// way to an object file
|
|
Module *importedFrom;
|
|
|
|
Dsymbols *decldefs; // top level declarations for this Module
|
|
|
|
Modules aimports; // all imported modules
|
|
|
|
unsigned debuglevel; // debug level
|
|
Strings *debugids; // debug identifiers
|
|
Strings *debugidsNot; // forward referenced debug identifiers
|
|
|
|
unsigned versionlevel; // version level
|
|
Strings *versionids; // version identifiers
|
|
Strings *versionidsNot; // forward referenced version identifiers
|
|
|
|
Macro *macrotable; // document comment macros
|
|
Escape *escapetable; // document comment escapes
|
|
|
|
size_t nameoffset; // offset of module name from start of ModuleInfo
|
|
size_t namelen; // length of module name in characters
|
|
|
|
#if IN_LLVM
|
|
int32_t doDocComment; // enable generating doc comments for this module
|
|
int32_t doHdrGen; // enable generating header file for this module
|
|
#endif
|
|
|
|
Module(const char *arg, Identifier *ident, int doDocComment, int doHdrGen);
|
|
static Module* create(const char *arg, Identifier *ident, int doDocComment, int doHdrGen);
|
|
|
|
static Module *load(Loc loc, Identifiers *packages, Identifier *ident);
|
|
|
|
const char *kind();
|
|
File *setOutfile(const char *name, const char *dir, const char *arg, const char *ext);
|
|
void setDocfile();
|
|
bool read(Loc loc); // read file, returns 'true' if succeed, 'false' otherwise.
|
|
#if IN_LLVM
|
|
Module *parse(bool gen_docs = false); // syntactic parse
|
|
#else
|
|
Module *parse(); // syntactic parse
|
|
#endif
|
|
void importAll(Scope *sc);
|
|
void semantic(Scope *); // semantic analysis
|
|
void semantic2(Scope *); // pass 2 semantic analysis
|
|
void semantic3(Scope *); // pass 3 semantic analysis
|
|
int needModuleInfo();
|
|
Dsymbol *search(Loc loc, Identifier *ident, int flags = SearchLocalsOnly);
|
|
Dsymbol *symtabInsert(Dsymbol *s);
|
|
void deleteObjFile();
|
|
static void addDeferredSemantic(Dsymbol *s);
|
|
static void runDeferredSemantic();
|
|
static void addDeferredSemantic3(Dsymbol *s);
|
|
static void runDeferredSemantic3();
|
|
static void clearCache();
|
|
int imports(Module *m);
|
|
|
|
bool isRoot() { return this->importedFrom == this; }
|
|
// true if the module source file is directly
|
|
// listed in command line.
|
|
bool isCoreModule(Identifier *ident);
|
|
|
|
// Back end
|
|
|
|
int doppelganger; // sub-module
|
|
Symbol *cov; // private uint[] __coverage;
|
|
unsigned *covb; // bit array of valid code line numbers
|
|
|
|
Symbol *sictor; // module order independent constructor
|
|
Symbol *sctor; // module constructor
|
|
Symbol *sdtor; // module destructor
|
|
Symbol *ssharedctor; // module shared constructor
|
|
Symbol *sshareddtor; // module shared destructor
|
|
Symbol *stest; // module unit test
|
|
|
|
Symbol *sfilename; // symbol for filename
|
|
|
|
Symbol *massert; // module assert function
|
|
Symbol *munittest; // module unittest failure function
|
|
Symbol *marray; // module array bounds function
|
|
|
|
#if IN_LLVM
|
|
// LDC
|
|
llvm::Module* genLLVMModule(llvm::LLVMContext& context);
|
|
File* buildFilePath(const char* forcename, const char* path, const char* ext, bool preservePaths, bool fqnNames);
|
|
|
|
bool llvmForceLogging;
|
|
bool noModuleInfo; /// Do not emit any module metadata.
|
|
|
|
// array ops emitted in this module already
|
|
AA *arrayfuncs;
|
|
|
|
// Coverage analysis
|
|
llvm::GlobalVariable* d_cover_valid; // private immutable size_t[] _d_cover_valid;
|
|
llvm::GlobalVariable* d_cover_data; // private uint[] _d_cover_data;
|
|
Array<size_t> d_cover_valid_init; // initializer for _d_cover_valid
|
|
#endif
|
|
|
|
Module *isModule() { return this; }
|
|
void accept(Visitor *v) { v->visit(this); }
|
|
};
|
|
|
|
struct ModuleDeclaration
|
|
{
|
|
Loc loc;
|
|
Identifier *id;
|
|
Identifiers *packages; // array of Identifier's representing packages
|
|
bool isdeprecated; // if it is a deprecated module
|
|
Expression *msg;
|
|
|
|
ModuleDeclaration(Loc loc, Identifiers *packages, Identifier *id);
|
|
|
|
const char *toChars();
|
|
};
|
|
|
|
#endif /* DMD_MODULE_H */
|