| /** |
| * Defines a package and module. |
| * |
| * Specification: $(LINK2 https://dlang.org/spec/module.html, Modules) |
| * |
| * Copyright: Copyright (C) 1999-2025 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/compiler/src/dmd/dmodule.d, _dmodule.d) |
| * Documentation: https://dlang.org/phobos/dmd_dmodule.html |
| * Coverage: https://codecov.io/gh/dlang/dmd/src/master/compiler/src/dmd/dmodule.d |
| */ |
| |
| module dmd.dmodule; |
| |
| import core.stdc.stdio; |
| import core.stdc.stdlib; |
| import core.stdc.string; |
| |
| import dmd.aggregate; |
| import dmd.arraytypes; |
| import dmd.astcodegen; |
| import dmd.astenums; |
| import dmd.common.outbuffer; |
| import dmd.compiler; |
| import dmd.cparse; |
| import dmd.declaration; |
| import dmd.dimport; |
| import dmd.dmacro; |
| import dmd.doc; |
| import dmd.dscope; |
| import dmd.dsymbol; |
| import dmd.dsymbolsem : dsymbolSemantic, importAll, load, include; |
| import dmd.errors; |
| import dmd.errorsink; |
| import dmd.expression; |
| import dmd.expressionsem; |
| import dmd.file_manager; |
| import dmd.func; |
| import dmd.globals; |
| import dmd.gluelayer; |
| import dmd.id; |
| import dmd.identifier; |
| import dmd.location; |
| import dmd.parse; |
| import dmd.root.array; |
| import dmd.root.file; |
| import dmd.root.filename; |
| import dmd.root.port; |
| import dmd.root.rmem; |
| import dmd.root.string; |
| import dmd.rootobject; |
| import dmd.semantic2; |
| import dmd.semantic3; |
| import dmd.target; |
| import dmd.utils; |
| import dmd.visitor; |
| |
| version (Windows) |
| { |
| enum PathSeparator = '\\'; |
| } |
| else version (Posix) |
| { |
| enum PathSeparator = '/'; |
| } |
| else |
| static assert(0); |
| |
| version (IN_GCC) {} |
| else version (IN_LLVM) {} |
| else version = MARS; |
| |
| // function used to call semantic3 on a module's dependencies |
| void semantic3OnDependencies(Module m) |
| { |
| if (!m) |
| return; |
| |
| if (m.semanticRun > PASS.semantic3) |
| return; |
| |
| m.semantic3(null); |
| |
| foreach (i; 1 .. m.aimports.length) |
| semantic3OnDependencies(m.aimports[i]); |
| } |
| |
| /** |
| * Remove generated .di files on error and exit |
| */ |
| void removeHdrFilesAndFail(ref Param params, ref Modules modules) nothrow |
| { |
| if (params.dihdr.doOutput) |
| { |
| foreach (m; modules) |
| { |
| if (m.filetype == FileType.dhdr) |
| continue; |
| File.remove(m.hdrfile.toChars()); |
| } |
| } |
| |
| fatal(); |
| } |
| |
| /** |
| * Converts a chain of identifiers to the filename of the module |
| * |
| * Params: |
| * packages = the names of the "parent" packages |
| * ident = the name of the child package or module |
| * |
| * Returns: |
| * the filename of the child package or module |
| */ |
| private const(char)[] getFilename(Identifier[] packages, Identifier ident) nothrow |
| { |
| const(char)[] filename = ident.toString(); |
| |
| OutBuffer buf; |
| OutBuffer dotmods; |
| auto modAliases = &global.params.modFileAliasStrings; |
| |
| if (packages.length == 0 && modAliases.length == 0) |
| return filename; |
| |
| void checkModFileAlias(const(char)[] p) |
| { |
| /* Check and replace the contents of buf[] with |
| * an alias string from global.params.modFileAliasStrings[] |
| */ |
| dotmods.writestring(p); |
| foreach_reverse (const m; *modAliases) |
| { |
| const q = strchr(m, '='); |
| assert(q); |
| if (dotmods.length == q - m && memcmp(dotmods.peekChars(), m, q - m) == 0) |
| { |
| buf.setsize(0); |
| auto rhs = q[1 .. strlen(q)]; |
| if (rhs.length > 0 && (rhs[$ - 1] == '/' || rhs[$ - 1] == '\\')) |
| rhs = rhs[0 .. $ - 1]; // remove trailing separator |
| buf.writestring(rhs); |
| break; // last matching entry in ms[] wins |
| } |
| } |
| dotmods.writeByte('.'); |
| } |
| |
| foreach (pid; packages) |
| { |
| const p = pid.toString(); |
| buf.writestring(p); |
| if (modAliases.length) |
| checkModFileAlias(p); |
| buf.writeByte(PathSeparator); |
| } |
| buf.writestring(filename); |
| if (modAliases.length) |
| checkModFileAlias(filename); |
| buf.writeByte(0); |
| filename = buf.extractSlice()[0 .. $ - 1]; |
| |
| return filename; |
| } |
| |
| /*********************************************************** |
| */ |
| extern (C++) class Package : ScopeDsymbol |
| { |
| PKG isPkgMod = PKG.unknown; |
| uint tag; // auto incremented tag, used to mask package tree in scopes |
| Module mod; // !=null if isPkgMod == PKG.module_ |
| |
| final extern (D) this(Loc loc, Identifier ident) nothrow |
| { |
| super(loc, ident); |
| __gshared uint packageTag; |
| this.tag = packageTag++; |
| this.dsym = DSYM.package_; |
| } |
| |
| override const(char)* kind() const nothrow |
| { |
| return "package"; |
| } |
| |
| override bool equals(const RootObject o) const |
| { |
| // custom 'equals' for bug 17441. "package a" and "module a" are not equal |
| if (this == o) |
| return true; |
| auto p = cast(Package)o; |
| return p && isModule() == p.isModule() && ident.equals(p.ident); |
| } |
| |
| /**************************************************** |
| * Input: |
| * packages[] the pkg1.pkg2 of pkg1.pkg2.mod |
| * Returns: |
| * the symbol table that mod should be inserted into |
| * Output: |
| * *pparent the rightmost package, i.e. pkg2, or NULL if no packages |
| * *ppkg the leftmost package, i.e. pkg1, or NULL if no packages |
| */ |
| extern (D) static DsymbolTable resolve(Identifier[] packages, Dsymbol* pparent, Package* ppkg) |
| { |
| DsymbolTable dst = Module.modules; |
| Dsymbol parent = null; |
| //printf("Package::resolve()\n"); |
| if (ppkg) |
| *ppkg = null; |
| foreach (pid; packages) |
| { |
| Package pkg; |
| Dsymbol p = dst.lookup(pid); |
| if (!p) |
| { |
| pkg = new Package(Loc.initial, pid); |
| dst.insert(pkg); |
| pkg.parent = parent; |
| pkg.symtab = new DsymbolTable(); |
| } |
| else |
| { |
| pkg = p.isPackage(); |
| assert(pkg); |
| // It might already be a module, not a package, but that needs |
| // to be checked at a higher level, where a nice error message |
| // can be generated. |
| // dot net needs modules and packages with same name |
| // But we still need a symbol table for it |
| if (!pkg.symtab) |
| pkg.symtab = new DsymbolTable(); |
| } |
| parent = pkg; |
| dst = pkg.symtab; |
| if (ppkg && !*ppkg) |
| *ppkg = pkg; |
| if (pkg.isModule()) |
| { |
| // Return the module so that a nice error message can be generated |
| if (ppkg) |
| *ppkg = cast(Package)p; |
| break; |
| } |
| } |
| |
| if (pparent) |
| *pparent = parent; |
| return dst; |
| } |
| |
| /** |
| * Checks if pkg is a sub-package of this |
| * |
| * For example, if this qualifies to 'a1.a2' and pkg - to 'a1.a2.a3', |
| * this function returns 'true'. If it is other way around or qualified |
| * package paths conflict function returns 'false'. |
| * |
| * Params: |
| * pkg = possible subpackage |
| * |
| * Returns: |
| * see description |
| */ |
| final bool isAncestorPackageOf(const Package pkg) const |
| { |
| if (this == pkg) |
| return true; |
| if (!pkg || !pkg.parent) |
| return false; |
| return isAncestorPackageOf(pkg.parent.isPackage()); |
| } |
| |
| override void accept(Visitor v) |
| { |
| v.visit(this); |
| } |
| |
| final Module isPackageMod() |
| { |
| if (isPkgMod == PKG.module_) |
| { |
| return mod; |
| } |
| return null; |
| } |
| |
| /** |
| * Checks for the existence of a package.d to set isPkgMod appropriately |
| * if isPkgMod == PKG.unknown |
| */ |
| final void resolvePKGunknown() |
| { |
| if (isModule()) |
| return; |
| if (isPkgMod != PKG.unknown) |
| return; |
| |
| Identifier[] packages; |
| for (Dsymbol s = this.parent; s; s = s.parent) |
| packages ~= s.ident; |
| reverse(packages); |
| |
| ImportPathInfo pathThatFoundThis; |
| if (Module.find(getFilename(packages, ident), pathThatFoundThis)) |
| Module.load(Loc.initial, packages, this.ident, pathThatFoundThis); |
| else |
| isPkgMod = PKG.package_; |
| } |
| } |
| |
| /*********************************************************** |
| */ |
| extern (C++) final class Module : Package |
| { |
| extern (C++) __gshared Module rootModule; |
| extern (C++) __gshared DsymbolTable modules; // symbol table of all modules |
| extern (C++) __gshared Modules amodules; // array of all modules |
| extern (C++) __gshared Dsymbols deferred; // deferred Dsymbol's needing semantic() run on them |
| extern (C++) __gshared Dsymbols deferred2; // deferred Dsymbol's needing semantic2() run on them |
| extern (C++) __gshared Dsymbols deferred3; // deferred Dsymbol's needing semantic3() run on them |
| |
| static void _init() |
| { |
| modules = new DsymbolTable(); |
| } |
| |
| /** |
| * Deinitializes the global state of the compiler. |
| * |
| * This can be used to restore the state set by `_init` to its original |
| * state. |
| */ |
| static void deinitialize() |
| { |
| modules = modules.init; |
| } |
| |
| extern (C++) __gshared AggregateDeclaration moduleinfo; |
| |
| const(char)[] arg; // original argument name |
| ModuleDeclaration* md; // if !=null, the contents of the ModuleDeclaration declaration |
| const FileName srcfile; // input source file |
| FileName objfile; // output .obj file |
| FileName hdrfile; // 'header' file |
| FileName docfile; // output documentation file |
| const(ubyte)[] src; /// Raw content of the file |
| uint errors; // if any errors in file |
| uint numlines; // number of lines in source file |
| FileType filetype; // source file type |
| bool hasAlwaysInlines; // contains references to functions that must be inlined |
| bool isPackageFile; // if it is a package.d |
| Edition edition; // language edition that this module is compiled with |
| Package pkg; // if isPackageFile is true, the Package that contains this package.d |
| Strings contentImportedFiles; // array of files whose content was imported |
| int needmoduleinfo; |
| private ThreeState selfimports; |
| private ThreeState rootimports; |
| Dsymbol[void*] tagSymTab; /// ImportC: tag symbols that conflict with other symbols used as the index |
| |
| private OutBuffer defines; // collect all the #define lines here |
| |
| |
| /************************************* |
| * Return true if module imports itself. |
| */ |
| bool selfImports() |
| { |
| //printf("Module::selfImports() %s\n", toChars()); |
| if (selfimports == ThreeState.none) |
| { |
| foreach (Module m; amodules) |
| m.insearch = false; |
| selfimports = imports(this) ? ThreeState.yes : ThreeState.no; |
| foreach (Module m; amodules) |
| m.insearch = false; |
| } |
| return selfimports == ThreeState.yes; |
| } |
| |
| /************************************* |
| * Return true if module imports root module. |
| */ |
| bool rootImports() |
| { |
| //printf("Module::rootImports() %s\n", toChars()); |
| if (rootimports == ThreeState.none) |
| { |
| foreach (Module m; amodules) |
| m.insearch = false; |
| rootimports = ThreeState.no; |
| foreach (Module m; amodules) |
| { |
| if (m.isRoot() && imports(m)) |
| { |
| rootimports = ThreeState.yes; |
| break; |
| } |
| } |
| foreach (Module m; amodules) |
| m.insearch = false; |
| } |
| return rootimports == ThreeState.yes; |
| } |
| |
| Identifier searchCacheIdent; |
| Dsymbol searchCacheSymbol; // cached value of search |
| SearchOptFlags searchCacheFlags; // cached flags |
| bool insearch; |
| |
| /** |
| * A root module is one that will be compiled all the way to |
| * object code. This field holds the root module that caused |
| * this module to be loaded. If this module is a root module, |
| * then it will be set to `this`. This is used to determine |
| * ownership of template instantiation. |
| */ |
| Module importedFrom; |
| |
| Dsymbols* decldefs; // top level declarations for this Module |
| |
| Modules aimports; // all imported modules |
| |
| Identifiers* debugids; // debug identifiers |
| Identifiers* debugidsNot; // forward referenced debug identifiers |
| |
| Identifiers* versionids; // version identifiers |
| Identifiers* versionidsNot; // forward referenced version identifiers |
| |
| MacroTable 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 |
| |
| extern (D) this(Loc loc, const(char)[] filename, Identifier ident, int doDocComment, int doHdrGen) |
| { |
| super(loc, ident); |
| this.dsym = DSYM.module_; |
| const(char)[] srcfilename; |
| //printf("Module::Module(filename = '%.*s', ident = '%s')\n", cast(int)filename.length, filename.ptr, ident.toChars()); |
| this.arg = filename; |
| srcfilename = FileName.defaultExt(filename, mars_ext); |
| if (target.run_noext && global.params.run && |
| !FileName.ext(filename) && |
| FileName.exists(srcfilename) == 0 && |
| FileName.exists(filename) == 1) |
| { |
| FileName.free(srcfilename.ptr); |
| srcfilename = FileName.sansExt(filename); |
| } |
| else if (!FileName.equalsExt(srcfilename, mars_ext) && |
| !FileName.equalsExt(srcfilename, hdr_ext) && |
| !FileName.equalsExt(srcfilename, c_ext) && |
| !FileName.equalsExt(srcfilename, i_ext) && |
| !FileName.equalsExt(srcfilename, dd_ext)) |
| { |
| |
| error(loc, "%s `%s` source file name '%.*s' must have .%.*s extension", |
| kind, toPrettyChars, |
| cast(int)srcfilename.length, srcfilename.ptr, |
| cast(int)mars_ext.length, mars_ext.ptr); |
| fatal(); |
| } |
| |
| srcfile = FileName(srcfilename); |
| objfile = setOutfilename(global.params.objname, global.params.objdir, filename, target.obj_ext); |
| if (doDocComment) |
| setDocfile(); |
| if (doHdrGen) |
| hdrfile = setOutfilename(global.params.dihdr.name, global.params.dihdr.dir, arg, hdr_ext); |
| |
| this.edition = Edition.legacy; |
| } |
| |
| extern (D) this(const(char)[] filename, Identifier ident, int doDocComment, int doHdrGen) |
| { |
| this(Loc.initial, filename, ident, doDocComment, doHdrGen); |
| } |
| |
| static Module create(const(char)* filename, Identifier ident, int doDocComment, int doHdrGen) |
| { |
| return create(filename.toDString, ident, doDocComment, doHdrGen); |
| } |
| |
| extern (D) static Module create(const(char)[] filename, Identifier ident, int doDocComment, int doHdrGen) |
| { |
| return new Module(Loc.initial, filename, ident, doDocComment, doHdrGen); |
| } |
| |
| static const(char)* find(const(char)* filename) |
| { |
| ImportPathInfo pathThatFoundThis; // is this needed? In fact is this function needed still??? |
| return find(filename.toDString, pathThatFoundThis).ptr; |
| } |
| |
| extern (D) static const(char)[] find(const(char)[] filename, out ImportPathInfo pathThatFoundThis) |
| { |
| ptrdiff_t whichPathFoundThis; |
| const(char)[] ret = global.fileManager.lookForSourceFile(filename, global.path[], whichPathFoundThis); |
| |
| if (whichPathFoundThis >= 0) |
| pathThatFoundThis = global.path[whichPathFoundThis]; |
| return ret; |
| } |
| |
| extern (C++) static Module load(Loc loc, Identifiers* packages, Identifier ident) |
| { |
| return load(loc, packages ? (*packages)[] : null, ident); |
| } |
| |
| extern (D) static Module load(Loc loc, Identifier[] packages, Identifier ident, ImportPathInfo pathInfo = ImportPathInfo.init) |
| { |
| //printf("Module::load(ident = '%s')\n", ident.toChars()); |
| // Build module filename by turning: |
| // foo.bar.baz |
| // into: |
| // foo\bar\baz |
| const(char)[] filename = getFilename(packages, ident); |
| // Look for the source file |
| ImportPathInfo importPathThatFindUs; |
| if (const result = find(filename, importPathThatFindUs)) |
| { |
| filename = result; // leaks |
| pathInfo = importPathThatFindUs; |
| } |
| |
| auto m = new Module(loc, filename, ident, 0, 0); |
| |
| // TODO: apply import path information (pathInfo) on to module |
| |
| if (!m.read(loc)) |
| return null; |
| if (global.params.v.verbose) |
| { |
| OutBuffer buf; |
| foreach (pid; packages) |
| { |
| buf.writestring(pid.toString()); |
| buf.writeByte('.'); |
| } |
| buf.printf("%s\t(%s)", ident.toChars(), m.srcfile.toChars()); |
| message("import %s", buf.peekChars()); |
| } |
| if((m = m.parse()) is null) return null; |
| |
| return m; |
| } |
| |
| override const(char)* kind() const |
| { |
| return "module"; |
| } |
| |
| /********************************************* |
| * Combines things into output file name for .html and .di files. |
| * Input: |
| * name Command line name given for the file, NULL if none |
| * dir Command line directory given for the file, NULL if none |
| * arg Name of the source file |
| * ext File name extension to use if 'name' is NULL |
| * global.params.preservePaths get output path from arg |
| * srcfile Input file - output file name must not match input file |
| */ |
| extern(D) FileName setOutfilename(const(char)[] name, const(char)[] dir, const(char)[] arg, const(char)[] ext) |
| { |
| const(char)[] docfilename; |
| if (name) |
| { |
| docfilename = name; |
| } |
| else |
| { |
| const(char)[] argdoc; |
| if (global.params.preservePaths) |
| argdoc = arg; |
| else |
| argdoc = FileName.name(arg); |
| |
| if (global.params.fullyQualifiedObjectFiles) |
| { |
| const fqn = md ? md.toString() : toString(); |
| argdoc = FileName.replaceName(argdoc, fqn); |
| |
| // add ext, otherwise forceExt will make nested.module into nested.<ext> |
| const bufferLength = argdoc.length + 1 + ext.length + /* null terminator */ 1; |
| char[] s = new char[bufferLength]; |
| s[0 .. argdoc.length] = argdoc[]; |
| s[argdoc.length] = '.'; |
| s[$-1-ext.length .. $-1] = ext[]; |
| s[$-1] = 0; |
| argdoc = s; |
| } |
| |
| // If argdoc doesn't have an absolute path, make it relative to dir |
| if (!FileName.absolute(argdoc)) |
| { |
| //FileName::ensurePathExists(dir); |
| argdoc = FileName.combine(dir, argdoc); |
| } |
| docfilename = FileName.forceExt(argdoc, ext); |
| } |
| if (FileName.equals(docfilename, srcfile.toString())) |
| { |
| error(loc, "%s `%s` source file and output file have same name '%s'", |
| kind, toPrettyChars, srcfile.toChars()); |
| fatal(); |
| } |
| return FileName(docfilename); |
| } |
| |
| extern (D) void setDocfile() |
| { |
| docfile = setOutfilename(global.params.ddoc.name, global.params.ddoc.dir, arg, doc_ext); |
| } |
| |
| /** |
| * Trigger the relevant semantic error when a file cannot be read |
| * |
| * We special case `object.d` as a failure is likely to be a rare |
| * but difficult to diagnose case for the user. Packages also require |
| * special handling to avoid exposing the compiler's internals. |
| * |
| * Params: |
| * loc = The location at which the file read originated (e.g. import) |
| */ |
| private void onFileReadError(Loc loc) |
| { |
| const name = srcfile.toString(); |
| if (FileName.equals(name, "object.d")) |
| { |
| ObjectNotFound(Loc.initial, ident); |
| } |
| else if (FileName.ext(this.arg) || !loc.isValid()) |
| { |
| // Modules whose original argument name has an extension, or do not |
| // have a valid location come from the command-line. |
| // Error that their file cannot be found and return early. |
| .error(loc, "cannot find input file `%.*s`", cast(int)name.length, name.ptr); |
| } |
| else |
| { |
| // if module is not named 'package' but we're trying to read 'package.d', we're looking for a package module |
| bool isPackageMod = (strcmp(toChars(), "package") != 0) && isPackageFileName(srcfile); |
| if (isPackageMod) |
| .error(loc, "importing package '%s' requires a 'package.d' file which cannot be found in '%.*s'", toChars(), cast(int)name.length, name.ptr); |
| else |
| { |
| .error(loc, "unable to read module `%s`", toChars()); |
| const pkgfile = FileName.combine(FileName.sansExt(name), package_d); |
| .errorSupplemental(loc, "Expected '%.*s' or '%.*s' in one of the following import paths:", |
| cast(int)name.length, name.ptr, cast(int)pkgfile.length, pkgfile.ptr); |
| } |
| } |
| if (!global.gag) |
| { |
| /* Print path |
| */ |
| if (global.path.length) |
| { |
| foreach (i, p; global.path[]) |
| fprintf(stderr, "import path[%llu] = %s\n", cast(ulong)i, p.path); |
| } |
| else |
| { |
| fprintf(stderr, "Specify path to file '%.*s' with -I switch\n", cast(int)name.length, name.ptr); |
| } |
| |
| removeHdrFilesAndFail(global.params, Module.amodules); |
| } |
| } |
| |
| /** |
| * Reads the file from `srcfile` and loads the source buffer. |
| * |
| * If makefile module dependency is requested, we add this module |
| * to the list of dependencies from here. |
| * |
| * Params: |
| * loc = the location |
| * |
| * Returns: `true` if successful |
| */ |
| bool read(Loc loc) |
| { |
| if (this.src) |
| return true; // already read |
| |
| //printf("Module::read('%s') file '%s'\n", toChars(), srcfile.toChars()); |
| |
| /* Preprocess the file if it's a .c file |
| */ |
| FileName filename = srcfile; |
| |
| const(ubyte)[] srctext; |
| if (global.preprocess && |
| FileName.equalsExt(srcfile.toString(), c_ext) && |
| FileName.exists(srcfile.toString())) |
| { |
| /* Apply C preprocessor to the .c file, returning the contents |
| * after preprocessing |
| */ |
| srctext = global.preprocess(srcfile, loc, defines).data; |
| } |
| else |
| srctext = global.fileManager.getFileContents(filename); |
| this.src = srctext; |
| |
| if (srctext) |
| { |
| if (global.params.makeDeps.doOutput) |
| global.params.makeDeps.files.push(srcfile.toChars()); |
| return true; |
| } |
| |
| this.onFileReadError(loc); |
| return false; |
| } |
| |
| /// syntactic parse |
| Module parse() |
| { |
| return parseModule!ASTCodegen(); |
| } |
| |
| /// ditto |
| extern (D) Module parseModule(AST)() |
| { |
| const(char)* srcname = srcfile.toChars(); |
| //printf("Module::parse(srcname = '%s')\n", srcname); |
| |
| import dmd.timetrace; |
| timeTraceBeginEvent(TimeTraceEventType.parse); |
| scope (exit) timeTraceEndEvent(TimeTraceEventType.parse, this); |
| |
| isPackageFile = isPackageFileName(srcfile); |
| const(char)[] buf = processSource(src, this); |
| // an error happened on UTF conversion |
| if (buf is null) return null; |
| |
| /* If it starts with the string "Ddoc", then it's a documentation |
| * source file. |
| */ |
| if (buf.length>= 4 && buf[0..4] == "Ddoc") |
| { |
| comment = buf.ptr + 4; |
| filetype = FileType.ddoc; |
| if (!docfile) |
| setDocfile(); |
| return this; |
| } |
| /* If it has the extension ".dd", it is also a documentation |
| * source file. Documentation source files may begin with "Ddoc" |
| * but do not have to if they have the .dd extension. |
| * https://issues.dlang.org/show_bug.cgi?id=15465 |
| */ |
| if (FileName.equalsExt(arg, dd_ext)) |
| { |
| comment = buf.ptr; // the optional Ddoc, if present, is handled above. |
| filetype = FileType.ddoc; |
| if (!docfile) |
| setDocfile(); |
| return this; |
| } |
| /* If it has the extension ".di", it is a "header" file. |
| */ |
| if (FileName.equalsExt(arg, hdr_ext)) |
| filetype = FileType.dhdr; |
| |
| /// Promote `this` to a root module if requested via `-i` |
| void checkCompiledImport() |
| { |
| if (!this.isRoot() && Compiler.onImport(this)) |
| this.importedFrom = this; |
| } |
| |
| DsymbolTable dst; |
| Package ppack = null; |
| |
| /* If it has the extension ".c", it is a "C" file. |
| * If it has the extension ".i", it is a preprocessed "C" file. |
| */ |
| if (FileName.equalsExt(arg, c_ext) || FileName.equalsExt(arg, i_ext)) |
| { |
| filetype = FileType.c; |
| |
| global.compileEnv.masm = target.os == Target.OS.Windows; // Microsoft inline assembler format |
| scope p = new CParser!AST(this, buf, cast(bool) docfile, global.errorSink, target.c, &defines, &global.compileEnv); |
| global.compileEnv.masm = false; |
| p.nextToken(); |
| checkCompiledImport(); |
| members = p.parseModule(); |
| assert(!p.md); // C doesn't have module declarations |
| numlines = p.linnum; |
| } |
| else |
| { |
| const bool doUnittests = global.params.parsingUnittestsRequired(); |
| scope p = new Parser!AST(this, buf, cast(bool) docfile, global.errorSink, &global.compileEnv, doUnittests); |
| p.nextToken(); |
| p.parseModuleDeclaration(); |
| md = p.md; |
| |
| if (md) |
| { |
| /* A ModuleDeclaration, md, was provided. |
| * The ModuleDeclaration sets the packages this module appears in, and |
| * the name of this module. |
| */ |
| this.ident = md.id; |
| dst = Package.resolve(md.packages, &this.parent, &ppack); |
| } |
| |
| // Done after parsing the module header because `module x.y.z` may override the file name |
| checkCompiledImport(); |
| |
| members = p.parseModuleContent(); |
| numlines = p.linnum; |
| } |
| |
| /* The symbol table into which the module is to be inserted. |
| */ |
| |
| if (md) |
| { |
| // Mark the package path as accessible from the current module |
| // https://issues.dlang.org/show_bug.cgi?id=21661 |
| // Code taken from Import.addPackageAccess() |
| if (md.packages.length > 0) |
| { |
| // module a.b.c.d; |
| auto p = ppack; // a |
| addAccessiblePackage(p, Visibility(Visibility.Kind.private_)); |
| foreach (id; md.packages[1 .. $]) // [b, c] |
| { |
| p = cast(Package) p.symtab.lookup(id); |
| if (p is null) |
| break; |
| addAccessiblePackage(p, Visibility(Visibility.Kind.private_)); |
| } |
| } |
| assert(dst); |
| Module m = ppack ? ppack.isModule() : null; |
| if (m && !isPackageFileName(m.srcfile)) |
| { |
| .error(md.loc, "package name '%s' conflicts with usage as a module name in file %s", ppack.toPrettyChars(), m.srcfile.toChars()); |
| } |
| } |
| else |
| { |
| /* The name of the module is set to the source file name. |
| * There are no packages. |
| */ |
| dst = modules; // and so this module goes into global module symbol table |
| /* Check to see if module name is a valid identifier |
| */ |
| if (!Identifier.isValidIdentifier(this.ident.toChars())) |
| error(loc, "%s `%s` has non-identifier characters in filename, use module declaration instead", kind, toPrettyChars); |
| } |
| // Insert module into the symbol table |
| Dsymbol s = this; |
| if (isPackageFile) |
| { |
| /* If the source tree is as follows: |
| * pkg/ |
| * +- package.d |
| * +- common.d |
| * the 'pkg' will be incorporated to the internal package tree in two ways: |
| * import pkg; |
| * and: |
| * import pkg.common; |
| * |
| * If both are used in one compilation, 'pkg' as a module (== pkg/package.d) |
| * and a package name 'pkg' will conflict each other. |
| * |
| * To avoid the conflict: |
| * 1. If preceding package name insertion had occurred by Package::resolve, |
| * reuse the previous wrapping 'Package' if it exists |
| * 2. Otherwise, 'package.d' wrapped by 'Package' is inserted to the internal tree in here. |
| * |
| * Then change Package::isPkgMod to PKG.module_ and set Package::mod. |
| * |
| * Note that the 'wrapping Package' is the Package that contains package.d and other submodules, |
| * the one inserted to the symbol table. |
| */ |
| auto ps = dst.lookup(ident); |
| Package p = ps ? ps.isPackage() : null; |
| if (p is null) |
| { |
| p = new Package(Loc.initial, ident); |
| p.tag = this.tag; // reuse the same package tag |
| p.symtab = new DsymbolTable(); |
| } |
| this.tag = p.tag; // reuse the 'older' package tag |
| this.pkg = p; |
| p.parent = this.parent; |
| p.isPkgMod = PKG.module_; |
| p.mod = this; |
| s = p; |
| } |
| if (!dst.insert(s)) |
| { |
| /* It conflicts with a name that is already in the symbol table. |
| * Figure out what went wrong, and issue error message. |
| */ |
| Dsymbol prev = dst.lookup(ident); |
| assert(prev); |
| if (Module mprev = prev.isModule()) |
| { |
| if (!FileName.equals(srcname, mprev.srcfile.toChars())) |
| error(loc, "%s `%s` from file %s conflicts with another module %s from file %s", kind, toPrettyChars, srcname, mprev.toChars(), mprev.srcfile.toChars()); |
| else if (isRoot() && mprev.isRoot()) |
| error(loc, "%s `%s` from file %s is specified twice on the command line", kind, toPrettyChars, srcname); |
| else |
| error(loc, "%s `%s` from file %s must be imported with 'import %s;'", kind, toPrettyChars, srcname, toPrettyChars()); |
| // https://issues.dlang.org/show_bug.cgi?id=14446 |
| // Return previously parsed module to avoid AST duplication ICE. |
| return mprev; |
| } |
| else if (Package pkg = prev.isPackage()) |
| { |
| // 'package.d' loaded after a previous 'Package' insertion |
| if (isPackageFile) |
| amodules.push(this); // Add to global array of all modules |
| else |
| error(md ? md.loc : loc, "%s `%s` from file %s conflicts with package name %s", kind, toPrettyChars, srcname, pkg.toChars()); |
| } |
| else |
| assert(global.errors); |
| } |
| else |
| { |
| // Add to global array of all modules |
| amodules.push(this); |
| } |
| Compiler.onParseModule(this); |
| return this; |
| } |
| |
| /********************************** |
| * Determine if we need to generate an instance of ModuleInfo |
| * for this Module. |
| */ |
| int needModuleInfo() |
| { |
| //printf("needModuleInfo() %s, %d, %d\n", toChars(), needmoduleinfo, global.params.cov); |
| return needmoduleinfo || global.params.cov; |
| } |
| |
| /******************************************* |
| * Print deprecation warning if we're deprecated, when |
| * this module is imported from scope sc. |
| * |
| * Params: |
| * sc = the scope into which we are imported |
| * loc = the location of the import statement |
| */ |
| void checkImportDeprecation(Loc loc, Scope* sc) |
| { |
| if (md && md.isdeprecated && !sc.isDeprecated) |
| { |
| Expression msg = md.msg; |
| if (StringExp se = msg ? msg.toStringExp() : null) |
| { |
| const slice = se.peekString(); |
| if (slice.length) |
| { |
| deprecation(loc, "%s `%s` is deprecated - %.*s", kind, toPrettyChars, cast(int)slice.length, slice.ptr); |
| return; |
| } |
| } |
| deprecation(loc, "%s `%s` is deprecated", kind, toPrettyChars); |
| } |
| } |
| |
| override bool isPackageAccessible(Package p, Visibility visibility, SearchOptFlags flags = SearchOpt.all) |
| { |
| if (insearch) // don't follow import cycles |
| return false; |
| insearch = true; |
| scope (exit) |
| insearch = false; |
| if (flags & SearchOpt.ignorePrivateImports) |
| visibility = Visibility(Visibility.Kind.public_); // only consider public imports |
| return super.isPackageAccessible(p, visibility); |
| } |
| |
| override Dsymbol symtabInsert(Dsymbol s) |
| { |
| searchCacheIdent = null; // symbol is inserted, so invalidate cache |
| return Package.symtabInsert(s); |
| } |
| |
| extern (D) void deleteObjFile() |
| { |
| if (global.params.obj) |
| File.remove(objfile.toChars()); |
| if (docfile) |
| File.remove(docfile.toChars()); |
| } |
| |
| /******************************************* |
| * Can't run semantic on s now, try again later. |
| */ |
| extern (D) static void addDeferredSemantic(Dsymbol s) |
| { |
| //printf("Module::addDeferredSemantic('%s')\n", s.toChars()); |
| if (!deferred.contains(s)) |
| deferred.push(s); |
| } |
| |
| extern (D) static void addDeferredSemantic2(Dsymbol s) |
| { |
| //printf("Module::addDeferredSemantic2('%s')\n", s.toChars()); |
| if (!deferred2.contains(s)) |
| deferred2.push(s); |
| } |
| |
| extern (D) static void addDeferredSemantic3(Dsymbol s) |
| { |
| //printf("Module::addDeferredSemantic3('%s')\n", s.toChars()); |
| if (!deferred.contains(s)) |
| deferred3.push(s); |
| } |
| |
| /****************************************** |
| * Run semantic() on deferred symbols. |
| */ |
| static void runDeferredSemantic() |
| { |
| __gshared int nested; |
| if (nested) |
| return; |
| //if (deferred.length) printf("+Module::runDeferredSemantic(), len = %ld\n", deferred.length); |
| nested++; |
| |
| size_t len; |
| do |
| { |
| len = deferred.length; |
| if (!len) |
| break; |
| |
| Dsymbol* todo; |
| Dsymbol* todoalloc = null; |
| Dsymbol tmp; |
| if (len == 1) |
| { |
| todo = &tmp; |
| } |
| else |
| { |
| todo = cast(Dsymbol*)Mem.check(malloc(len * Dsymbol.sizeof)); |
| todoalloc = todo; |
| } |
| memcpy(todo, deferred.tdata(), len * Dsymbol.sizeof); |
| deferred.setDim(0); |
| |
| foreach (i; 0..len) |
| { |
| Dsymbol s = todo[i]; |
| s.dsymbolSemantic(null); |
| //printf("deferred: %s, parent = %s\n", s.toChars(), s.parent.toChars()); |
| } |
| //printf("\tdeferred.length = %ld, len = %ld\n", deferred.length, len); |
| if (todoalloc) |
| free(todoalloc); |
| } |
| while (deferred.length != len); // while making progress |
| nested--; |
| //printf("-Module::runDeferredSemantic(), len = %ld\n", deferred.length); |
| } |
| |
| static void runDeferredSemantic2() |
| { |
| Module.runDeferredSemantic(); |
| |
| Dsymbols* a = &Module.deferred2; |
| for (size_t i = 0; i < a.length; i++) |
| { |
| Dsymbol s = (*a)[i]; |
| //printf("[%d] %s semantic2a\n", i, s.toPrettyChars()); |
| s.semantic2(null); |
| |
| if (global.errors) |
| break; |
| } |
| a.setDim(0); |
| } |
| |
| static void runDeferredSemantic3() |
| { |
| Module.runDeferredSemantic2(); |
| |
| Dsymbols* a = &Module.deferred3; |
| for (size_t i = 0; i < a.length; i++) |
| { |
| Dsymbol s = (*a)[i]; |
| //printf("[%d] %s semantic3a\n", i, s.toPrettyChars()); |
| s.semantic3(null); |
| |
| if (global.errors) |
| break; |
| } |
| a.setDim(0); |
| } |
| |
| extern (D) static void clearCache() nothrow |
| { |
| foreach (Module m; amodules) |
| m.searchCacheIdent = null; |
| } |
| |
| /************************************ |
| * Recursively look at every module this module imports, |
| * return true if it imports m. |
| * Can be used to detect circular imports. |
| */ |
| int imports(Module m) nothrow |
| { |
| //printf("%s Module::imports(%s)\n", toChars(), m.toChars()); |
| version (none) |
| { |
| foreach (i, Module mi; aimports) |
| printf("\t[%d] %s\n", cast(int) i, mi.toChars()); |
| } |
| foreach (Module mi; aimports) |
| { |
| if (mi == m) |
| return true; |
| if (!mi.insearch) |
| { |
| mi.insearch = true; |
| int r = mi.imports(m); |
| if (r) |
| return r; |
| } |
| } |
| return false; |
| } |
| |
| bool isRoot() nothrow |
| { |
| return this.importedFrom == this; |
| } |
| |
| /// Returns: Whether this module is in the `core` package and has name `ident` |
| bool isCoreModule(Identifier ident) nothrow |
| { |
| return this.ident == ident && parent && parent.ident == Id.core && !parent.parent; |
| } |
| |
| // Back end |
| int doppelganger; // sub-module |
| Symbol* cov; // private uint[] __coverage; |
| uint[] 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 |
| |
| uint[uint] ctfe_cov; /// coverage information from ctfe execution_count[line] |
| |
| override void accept(Visitor v) |
| { |
| v.visit(this); |
| } |
| |
| /*********************************************** |
| * Writes this module's fully-qualified name to buf |
| * Params: |
| * buf = The buffer to write to |
| */ |
| void fullyQualifiedName(ref OutBuffer buf) nothrow |
| { |
| buf.writestring(ident.toString()); |
| |
| for (auto package_ = parent; package_ !is null; package_ = package_.parent) |
| { |
| buf.prependstring("."); |
| buf.prependstring(package_.ident.toChars()); |
| } |
| } |
| |
| /** Lazily initializes and returns the escape table. |
| Turns out it eats a lot of memory. |
| */ |
| extern(D) Escape* escapetable() nothrow |
| { |
| if (!_escapetable) |
| _escapetable = new Escape(); |
| return _escapetable; |
| } |
| |
| /**************************** |
| * A Singleton that loads core.stdc.config |
| * Returns: |
| * Module of core.stdc.config, null if couldn't find it |
| */ |
| extern (D) static Module loadCoreStdcConfig() |
| { |
| __gshared Module core_stdc_config; |
| auto pkgids = new Identifier[2]; |
| pkgids[0] = Id.core; |
| pkgids[1] = Id.stdc; |
| return loadModuleFromLibrary(core_stdc_config, pkgids, Id.config); |
| } |
| |
| /**************************** |
| * A Singleton that loads core.atomic |
| * Returns: |
| * Module of core.atomic, null if couldn't find it |
| */ |
| extern (D) static Module loadCoreAtomic() |
| { |
| __gshared Module core_atomic; |
| auto pkgids = new Identifier[1]; |
| pkgids[0] = Id.core; |
| return loadModuleFromLibrary(core_atomic, pkgids, Id.atomic); |
| } |
| |
| /**************************** |
| * A Singleton that loads std.math |
| * Returns: |
| * Module of std.math, null if couldn't find it |
| */ |
| extern (D) static Module loadStdMath() |
| { |
| __gshared Module std_math; |
| auto pkgids = new Identifier[1]; |
| pkgids[0] = Id.std; |
| return loadModuleFromLibrary(std_math, pkgids, Id.math); |
| } |
| |
| /********************************** |
| * Load a Module from the library. |
| * Params: |
| * mod = cached return value of this call |
| * pkgids = package identifiers |
| * modid = module id |
| * Returns: |
| * Module loaded, null if cannot load it |
| */ |
| extern (D) private static Module loadModuleFromLibrary(ref Module mod, Identifier[] pkgids, Identifier modid) |
| { |
| if (mod) |
| return mod; |
| |
| auto imp = new Import(Loc.initial, pkgids[], modid, null, true); |
| // Module.load will call fatal() if there's no module available. |
| // Gag the error here, pushing the error handling to the caller. |
| const errors = global.startGagging(); |
| imp.load(null); |
| if (imp.mod) |
| { |
| imp.mod.importAll(null); |
| imp.mod.dsymbolSemantic(null); |
| } |
| global.endGagging(errors); |
| mod = imp.mod; |
| return mod; |
| } |
| } |
| |
| /*********************************************************** |
| */ |
| extern (C++) struct ModuleDeclaration |
| { |
| Loc loc; |
| Identifier id; |
| Identifier[] packages; // array of Identifier's representing packages |
| bool isdeprecated; // if it is a deprecated module |
| Expression msg; |
| |
| extern (D) this(Loc loc, Identifier[] packages, Identifier id, Expression msg, bool isdeprecated) @safe |
| { |
| this.loc = loc; |
| this.packages = packages; |
| this.id = id; |
| this.msg = msg; |
| this.isdeprecated = isdeprecated; |
| } |
| |
| extern (C++) const(char)* toChars() const @safe |
| { |
| OutBuffer buf; |
| foreach (pid; packages) |
| { |
| buf.writestring(pid.toString()); |
| buf.writeByte('.'); |
| } |
| buf.writestring(id.toString()); |
| return buf.extractChars(); |
| } |
| |
| /// Provide a human readable representation |
| extern (D) const(char)[] toString() const |
| { |
| return this.toChars().toDString; |
| } |
| } |
| |
| /**************************************** |
| * Create array of the local classes in the Module, suitable |
| * for inclusion in ModuleInfo |
| * Params: |
| * mod = the Module |
| * aclasses = array to fill in |
| * Returns: array of local classes |
| */ |
| void getLocalClasses(Module mod, ref ClassDeclarations aclasses) |
| { |
| //printf("members.length = %d\n", mod.members.length); |
| int pushAddClassDg(size_t n, Dsymbol sm) |
| { |
| if (!sm) |
| return 0; |
| |
| if (auto cd = sm.isClassDeclaration()) |
| { |
| // compatibility with previous algorithm |
| if (cd.parent && cd.parent.isTemplateMixin()) |
| return 0; |
| |
| if (cd.classKind != ClassKind.objc) |
| aclasses.push(cd); |
| } |
| return 0; |
| } |
| |
| _foreach(null, mod.members, &pushAddClassDg); |
| } |
| |
| |
| alias ForeachDg = int delegate(size_t idx, Dsymbol s); |
| |
| /*************************************** |
| * Expands attribute declarations in members in depth first |
| * order. Calls dg(size_t symidx, Dsymbol *sym) for each |
| * member. |
| * If dg returns !=0, stops and returns that value else returns 0. |
| * Use this function to avoid the O(N + N^2/2) complexity of |
| * calculating dim and calling N times getNth. |
| * Returns: |
| * last value returned by dg() |
| */ |
| int _foreach(Scope* sc, Dsymbols* members, scope ForeachDg dg, size_t* pn = null) |
| { |
| assert(dg); |
| if (!members) |
| return 0; |
| size_t n = pn ? *pn : 0; // take over index |
| int result = 0; |
| foreach (size_t i; 0 .. members.length) |
| { |
| import dmd.attrib : AttribDeclaration; |
| import dmd.dtemplate : TemplateMixin; |
| |
| Dsymbol s = (*members)[i]; |
| if (AttribDeclaration a = s.isAttribDeclaration()) |
| result = _foreach(sc, a.include(sc), dg, &n); |
| else if (TemplateMixin tm = s.isTemplateMixin()) |
| result = _foreach(sc, tm.members, dg, &n); |
| else if (s.isTemplateInstance()) |
| { |
| } |
| else if (s.isUnitTestDeclaration()) |
| { |
| } |
| else |
| result = dg(n++, s); |
| if (result) |
| break; |
| } |
| if (pn) |
| *pn = n; // update index |
| return result; |
| } |
| |
| /** |
| * Process the content of a source file |
| * |
| * Attempts to find which encoding it is using, if it has BOM, |
| * and then normalize the source to UTF-8. If no encoding is required, |
| * a slice of `src` will be returned without extra allocation. |
| * |
| * Params: |
| * src = Content of the source file to process |
| * mod = Module matching `src`, used for error handling |
| * |
| * Returns: |
| * UTF-8 encoded variant of `src`, stripped of any BOM, |
| * or `null` if an error happened. |
| */ |
| private const(char)[] processSource (const(ubyte)[] src, Module mod) |
| { |
| enum SourceEncoding { utf16, utf32} |
| enum Endian { little, big} |
| immutable loc = mod.loc; |
| |
| /* |
| * Convert a buffer from UTF32 to UTF8 |
| * Params: |
| * Endian = is the buffer big/little endian |
| * buf = buffer of UTF32 data |
| * Returns: |
| * input buffer reencoded as UTF8 |
| */ |
| |
| char[] UTF32ToUTF8(Endian endian)(const(char)[] buf) |
| { |
| static if (endian == Endian.little) |
| alias readNext = Port.readlongLE; |
| else |
| alias readNext = Port.readlongBE; |
| |
| if (buf.length & 3) |
| { |
| .error(loc, "%s `%s` odd length of UTF-32 char source %llu", |
| mod.kind, mod.toPrettyChars, cast(ulong) buf.length); |
| return null; |
| } |
| |
| const (uint)[] eBuf = cast(const(uint)[])buf; |
| |
| OutBuffer dbuf; |
| dbuf.reserve(eBuf.length); |
| |
| foreach (i; 0 .. eBuf.length) |
| { |
| const u = readNext(&eBuf[i]); |
| if (u & ~0x7F) |
| { |
| if (u > 0x10FFFF) |
| { |
| .error(loc, "%s `%s` UTF-32 value %08x greater than 0x10FFFF", mod.kind, mod.toPrettyChars, u); |
| return null; |
| } |
| dbuf.writeUTF8(u); |
| } |
| else |
| dbuf.writeByte(u); |
| } |
| dbuf.writeByte(0); //add null terminator |
| return dbuf.extractSlice(); |
| } |
| |
| /* |
| * Convert a buffer from UTF16 to UTF8 |
| * Params: |
| * Endian = is the buffer big/little endian |
| * buf = buffer of UTF16 data |
| * Returns: |
| * input buffer reencoded as UTF8 |
| */ |
| |
| char[] UTF16ToUTF8(Endian endian)(const(char)[] buf) |
| { |
| static if (endian == Endian.little) |
| alias readNext = Port.readwordLE; |
| else |
| alias readNext = Port.readwordBE; |
| |
| if (buf.length & 1) |
| { |
| .error(loc, "%s `%s` odd length of UTF-16 char source %llu", mod.kind, mod.toPrettyChars, cast(ulong) buf.length); |
| return null; |
| } |
| |
| const (ushort)[] eBuf = cast(const(ushort)[])buf; |
| |
| OutBuffer dbuf; |
| dbuf.reserve(eBuf.length); |
| |
| //i will be incremented in the loop for high codepoints |
| foreach (ref i; 0 .. eBuf.length) |
| { |
| uint u = readNext(&eBuf[i]); |
| if (u & ~0x7F) |
| { |
| if (0xD800 <= u && u < 0xDC00) |
| { |
| i++; |
| if (i >= eBuf.length) |
| { |
| .error(loc, "%s `%s` surrogate UTF-16 high value %04x at end of file", mod.kind, mod.toPrettyChars, u); |
| return null; |
| } |
| const u2 = readNext(&eBuf[i]); |
| if (u2 < 0xDC00 || 0xE000 <= u2) |
| { |
| .error(loc, "%s `%s` surrogate UTF-16 low value %04x out of range", mod.kind, mod.toPrettyChars, u2); |
| return null; |
| } |
| u = (u - 0xD7C0) << 10; |
| u |= (u2 - 0xDC00); |
| } |
| else if (u >= 0xDC00 && u <= 0xDFFF) |
| { |
| .error(loc, "%s `%s` unpaired surrogate UTF-16 value %04x", mod.kind, mod.toPrettyChars, u); |
| return null; |
| } |
| else if (u == 0xFFFE || u == 0xFFFF) |
| { |
| .error(loc, "%s `%s` illegal UTF-16 value %04x", mod.kind, mod.toPrettyChars, u); |
| return null; |
| } |
| dbuf.writeUTF8(u); |
| } |
| else |
| dbuf.writeByte(u); |
| } |
| dbuf.writeByte(0); //add a terminating null byte |
| return dbuf.extractSlice(); |
| } |
| |
| const(char)[] buf = cast(const(char)[]) src; |
| |
| // Assume the buffer is from memory and has not be read from disk. Assume UTF-8. |
| if (buf.length < 2) |
| return buf; |
| |
| /* Convert all non-UTF-8 formats to UTF-8. |
| * BOM : https://www.unicode.org/faq/utf_bom.html |
| * 00 00 FE FF UTF-32BE, big-endian |
| * FF FE 00 00 UTF-32LE, little-endian |
| * FE FF UTF-16BE, big-endian |
| * FF FE UTF-16LE, little-endian |
| * EF BB BF UTF-8 |
| */ |
| if (buf[0] == 0xFF && buf[1] == 0xFE) |
| { |
| if (buf.length >= 4 && buf[2] == 0 && buf[3] == 0) |
| return UTF32ToUTF8!(Endian.little)(buf[4 .. $]); |
| return UTF16ToUTF8!(Endian.little)(buf[2 .. $]); |
| } |
| |
| if (buf[0] == 0xFE && buf[1] == 0xFF) |
| return UTF16ToUTF8!(Endian.big)(buf[2 .. $]); |
| |
| if (buf.length >= 4 && buf[0] == 0 && buf[1] == 0 && buf[2] == 0xFE && buf[3] == 0xFF) |
| return UTF32ToUTF8!(Endian.big)(buf[4 .. $]); |
| |
| if (buf.length >= 3 && buf[0] == 0xEF && buf[1] == 0xBB && buf[2] == 0xBF) |
| return buf[3 .. $]; |
| |
| /* There is no BOM. Make use of Arcane Jill's insight that |
| * the first char of D source must be ASCII to |
| * figure out the encoding. |
| */ |
| if (buf.length >= 4 && buf[1] == 0 && buf[2] == 0 && buf[3] == 0) |
| return UTF32ToUTF8!(Endian.little)(buf); |
| if (buf.length >= 4 && buf[0] == 0 && buf[1] == 0 && buf[2] == 0) |
| return UTF32ToUTF8!(Endian.big)(buf); |
| // try to check for UTF-16 |
| if (buf.length >= 2 && buf[1] == 0) |
| return UTF16ToUTF8!(Endian.little)(buf); |
| if (buf[0] == 0) |
| return UTF16ToUTF8!(Endian.big)(buf); |
| |
| // It's UTF-8 |
| if (buf[0] >= 0x80) |
| { |
| .error(loc, "%s `%s` source file must start with BOM or ASCII character, not \\x%02X", mod.kind, mod.toPrettyChars, buf[0]); |
| return null; |
| } |
| |
| return buf; |
| } |
| |
| /******************************************* |
| * Look for member of the form: |
| * const(MemberInfo)[] getMembers(string); |
| * Returns NULL if not found |
| */ |
| FuncDeclaration findGetMembers(ScopeDsymbol dsym) |
| { |
| import dmd.opover : search_function; |
| Dsymbol s = search_function(dsym, Id.getmembers); |
| FuncDeclaration fdx = s ? s.isFuncDeclaration() : null; |
| version (none) |
| { |
| // Finish |
| __gshared TypeFunction tfgetmembers; |
| if (!tfgetmembers) |
| { |
| Scope sc; |
| sc.eSink = global.errorSink; |
| auto parameters = new Parameters(); |
| Parameters* p = new Parameter(STC.in_, Type.tchar.constOf().arrayOf(), null, null); |
| parameters.push(p); |
| Type tret = null; |
| TypeFunction tf = new TypeFunction(parameters, tret, VarArg.none, LINK.d); |
| tfgetmembers = tf.dsymbolSemantic(Loc.initial, &sc).isTypeFunction(); |
| } |
| if (fdx) |
| fdx = fdx.overloadExactMatch(tfgetmembers); |
| } |
| if (fdx && fdx.isVirtual()) |
| fdx = null; |
| return fdx; |
| } |