123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173 |
- #
- #
- # The Nim Compiler
- # (c) Copyright 2015 Andreas Rumpf
- #
- # See the file "copying.txt", included in this
- # distribution, for details about the copyright.
- #
- ## Implements the module handling, including the caching of modules.
- import
- ast, astalgo, magicsys, msgs, options,
- idents, lexer, passes, syntaxes, llstream, modulegraphs, rod,
- lineinfos, pathutils, tables
- proc resetSystemArtifacts*(g: ModuleGraph) =
- magicsys.resetSysTypes(g)
- template getModuleIdent(graph: ModuleGraph, filename: AbsoluteFile): PIdent =
- getIdent(graph.cache, splitFile(filename).name)
- template packageId(): untyped {.dirty.} = ItemId(module: PackageModuleId, item: int32(fileIdx))
- proc getPackage(graph: ModuleGraph; fileIdx: FileIndex): PSym =
- ## returns package symbol (skPackage) for yet to be defined module for fileIdx
- let filename = AbsoluteFile toFullPath(graph.config, fileIdx)
- let name = getModuleIdent(graph, filename)
- let info = newLineInfo(fileIdx, 1, 1)
- let
- pck = getPackageName(graph.config, filename.string)
- pck2 = if pck.len > 0: pck else: "unknown"
- pack = getIdent(graph.cache, pck2)
- var packSym = graph.packageSyms.strTableGet(pack)
- if packSym == nil:
- packSym = newSym(skPackage, getIdent(graph.cache, pck2), packageId(), nil, info)
- initStrTable(packSym.tab)
- graph.packageSyms.strTableAdd(packSym)
- else:
- let existing = strTableGet(packSym.tab, name)
- if existing != nil and existing.info.fileIndex != info.fileIndex:
- when false:
- # we used to produce an error:
- localError(graph.config, info,
- "module names need to be unique per Nimble package; module clashes with " &
- toFullPath(graph.config, existing.info.fileIndex))
- else:
- # but starting with version 0.20 we now produce a fake Nimble package instead
- # to resolve the conflicts:
- let pck3 = fakePackageName(graph.config, filename)
- # this makes the new `packSym`'s owner be the original `packSym`
- packSym = newSym(skPackage, getIdent(graph.cache, pck3), packageId(), packSym, info)
- initStrTable(packSym.tab)
- graph.packageSyms.strTableAdd(packSym)
- result = packSym
- proc partialInitModule(result: PSym; graph: ModuleGraph; fileIdx: FileIndex; filename: AbsoluteFile) =
- let packSym = getPackage(graph, fileIdx)
- result.owner = packSym
- result.position = int fileIdx
- if int(fileIdx) >= graph.modules.len:
- setLen(graph.modules, int(fileIdx) + 1)
- graph.modules[result.position] = result
- initStrTable(result.tab)
- strTableAdd(result.tab, result) # a module knows itself
- strTableAdd(packSym.tab, result)
- proc newModule(graph: ModuleGraph; fileIdx: FileIndex): PSym =
- let filename = AbsoluteFile toFullPath(graph.config, fileIdx)
- # We cannot call ``newSym`` here, because we have to circumvent the ID
- # mechanism, which we do in order to assign each module a persistent ID.
- result = PSym(kind: skModule, itemId: ItemId(module: int32(fileIdx), item: 0'i32),
- name: getModuleIdent(graph, filename),
- info: newLineInfo(fileIdx, 1, 1))
- if not isNimIdentifier(result.name.s):
- rawMessage(graph.config, errGenerated, "invalid module name: " & result.name.s)
- partialInitModule(result, graph, fileIdx, filename)
- proc compileModule*(graph: ModuleGraph; fileIdx: FileIndex; flags: TSymFlags): PSym =
- var flags = flags
- if fileIdx == graph.config.projectMainIdx2: flags.incl sfMainModule
- result = graph.getModule(fileIdx)
- template processModuleAux =
- var s: PLLStream
- if sfMainModule in flags:
- if graph.config.projectIsStdin: s = stdin.llStreamOpen
- elif graph.config.projectIsCmd: s = llStreamOpen(graph.config.cmdInput)
- discard processModule(graph, result, idGeneratorFromModule(result), s)
- if result == nil:
- let filename = AbsoluteFile toFullPath(graph.config, fileIdx)
- result = loadModuleSym(graph, fileIdx, filename)
- if result == nil:
- result = newModule(graph, fileIdx)
- result.flags.incl flags
- registerModule(graph, result)
- else:
- partialInitModule(result, graph, fileIdx, filename)
- processModuleAux()
- elif graph.isDirty(result):
- result.flags.excl sfDirty
- # reset module fields:
- initStrTable(result.tab)
- result.ast = nil
- processModuleAux()
- graph.markClientsDirty(fileIdx)
- proc importModule*(graph: ModuleGraph; s: PSym, fileIdx: FileIndex): PSym =
- # this is called by the semantic checking phase
- assert graph.config != nil
- result = compileModule(graph, fileIdx, {})
- graph.addDep(s, fileIdx)
- # keep track of import relationships
- if graph.config.hcrOn:
- graph.importDeps.mgetOrPut(FileIndex(s.position), @[]).add(fileIdx)
- #if sfSystemModule in result.flags:
- # localError(result.info, errAttemptToRedefine, result.name.s)
- # restore the notes for outer module:
- graph.config.notes =
- if s.getnimblePkgId == graph.config.mainPackageId or isDefined(graph.config, "booting"): graph.config.mainPackageNotes
- else: graph.config.foreignPackageNotes
- proc includeModule*(graph: ModuleGraph; s: PSym, fileIdx: FileIndex): PNode =
- result = syntaxes.parseFile(fileIdx, graph.cache, graph.config)
- graph.addDep(s, fileIdx)
- graph.addIncludeDep(s.position.FileIndex, fileIdx)
- proc connectCallbacks*(graph: ModuleGraph) =
- graph.includeFileCallback = includeModule
- graph.importModuleCallback = importModule
- proc compileSystemModule*(graph: ModuleGraph) =
- if graph.systemModule == nil:
- connectCallbacks(graph)
- graph.config.m.systemFileIdx = fileInfoIdx(graph.config,
- graph.config.libpath / RelativeFile"system.nim")
- discard graph.compileModule(graph.config.m.systemFileIdx, {sfSystemModule})
- proc wantMainModule*(conf: ConfigRef) =
- if conf.projectFull.isEmpty:
- fatal(conf, newLineInfo(conf, AbsoluteFile(commandLineDesc), 1, 1), errGenerated,
- "command expects a filename")
- conf.projectMainIdx = fileInfoIdx(conf, addFileExt(conf.projectFull, NimExt))
- proc compileProject*(graph: ModuleGraph; projectFileIdx = InvalidFileIdx) =
- connectCallbacks(graph)
- let conf = graph.config
- wantMainModule(conf)
- let systemFileIdx = fileInfoIdx(conf, conf.libpath / RelativeFile"system.nim")
- let projectFile = if projectFileIdx == InvalidFileIdx: conf.projectMainIdx else: projectFileIdx
- conf.projectMainIdx2 = projectFile
- let packSym = getPackage(graph, projectFile)
- graph.config.mainPackageId = packSym.getnimblePkgId
- graph.importStack.add projectFile
- if projectFile == systemFileIdx:
- discard graph.compileModule(projectFile, {sfMainModule, sfSystemModule})
- else:
- graph.compileSystemModule()
- discard graph.compileModule(projectFile, {sfMainModule})
- proc makeModule*(graph: ModuleGraph; filename: AbsoluteFile): PSym =
- result = graph.newModule(fileInfoIdx(graph.config, filename))
- registerModule(graph, result)
- proc makeModule*(graph: ModuleGraph; filename: string): PSym =
- result = makeModule(graph, AbsoluteFile filename)
- proc makeStdinModule*(graph: ModuleGraph): PSym = graph.makeModule(AbsoluteFile"stdin")
|