pipelines.nim 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300
  1. import sem, cgen, modulegraphs, ast, llstream, parser, msgs,
  2. lineinfos, reorder, options, semdata, cgendata, modules, pathutils,
  3. packages, syntaxes, depends, vm, pragmas, idents, lookups, wordrecg,
  4. liftdestructors
  5. import pipelineutils
  6. import ../dist/checksums/src/checksums/sha1
  7. when not defined(leanCompiler):
  8. import jsgen, docgen2
  9. import std/[syncio, objectdollar, assertions, tables, strutils, strtabs]
  10. import renderer
  11. import ic/replayer
  12. proc setPipeLinePass*(graph: ModuleGraph; pass: PipelinePass) =
  13. graph.pipelinePass = pass
  14. proc processPipeline(graph: ModuleGraph; semNode: PNode; bModule: PPassContext): PNode =
  15. case graph.pipelinePass
  16. of CgenPass:
  17. result = semNode
  18. if bModule != nil:
  19. genTopLevelStmt(BModule(bModule), result)
  20. of JSgenPass:
  21. when not defined(leanCompiler):
  22. result = processJSCodeGen(bModule, semNode)
  23. of GenDependPass:
  24. result = addDotDependency(bModule, semNode)
  25. of SemPass:
  26. result = graph.emptyNode
  27. of Docgen2Pass, Docgen2TexPass:
  28. when not defined(leanCompiler):
  29. result = processNode(bModule, semNode)
  30. of Docgen2JsonPass:
  31. when not defined(leanCompiler):
  32. result = processNodeJson(bModule, semNode)
  33. of EvalPass, InterpreterPass:
  34. result = interpreterCode(bModule, semNode)
  35. of NonePass:
  36. doAssert false, "use setPipeLinePass to set a proper PipelinePass"
  37. proc processImplicitImports(graph: ModuleGraph; implicits: seq[string], nodeKind: TNodeKind,
  38. m: PSym, ctx: PContext, bModule: PPassContext, idgen: IdGenerator,
  39. ) =
  40. # XXX fixme this should actually be relative to the config file!
  41. let relativeTo = toFullPath(graph.config, m.info)
  42. for module in items(implicits):
  43. # implicit imports should not lead to a module importing itself
  44. if m.position != resolveMod(graph.config, module, relativeTo).int32:
  45. var importStmt = newNodeI(nodeKind, m.info)
  46. var str = newStrNode(nkStrLit, module)
  47. str.info = m.info
  48. importStmt.add str
  49. message(graph.config, importStmt.info, hintProcessingStmt, $idgen[])
  50. let semNode = semWithPContext(ctx, importStmt)
  51. if semNode == nil or processPipeline(graph, semNode, bModule) == nil:
  52. break
  53. proc prePass(c: PContext; n: PNode) =
  54. for son in n:
  55. if son.kind == nkPragma:
  56. for s in son:
  57. var key = if s.kind in nkPragmaCallKinds and s.len > 1: s[0] else: s
  58. if key.kind in {nkBracketExpr, nkCast} or key.kind notin nkIdentKinds:
  59. continue
  60. let ident = whichKeyword(considerQuotedIdent(c, key))
  61. case ident
  62. of wReorder:
  63. pragmaNoForward(c, s, flag = sfReorder)
  64. of wExperimental:
  65. if isTopLevel(c) and s.kind in nkPragmaCallKinds and s.len == 2:
  66. let name = c.semConstExpr(c, s[1])
  67. case name.kind
  68. of nkStrLit, nkRStrLit, nkTripleStrLit:
  69. try:
  70. let feature = parseEnum[Feature](name.strVal)
  71. if feature == codeReordering:
  72. c.features.incl feature
  73. c.module.flags.incl sfReorder
  74. except ValueError:
  75. discard
  76. else:
  77. discard
  78. else:
  79. discard
  80. proc processPipelineModule*(graph: ModuleGraph; module: PSym; idgen: IdGenerator;
  81. stream: PLLStream): bool =
  82. if graph.stopCompile(): return true
  83. var
  84. p: Parser = default(Parser)
  85. s: PLLStream
  86. fileIdx = module.fileIdx
  87. prepareConfigNotes(graph, module)
  88. let ctx = preparePContext(graph, module, idgen)
  89. let bModule: PPassContext =
  90. case graph.pipelinePass
  91. of CgenPass:
  92. setupCgen(graph, module, idgen)
  93. of JSgenPass:
  94. when not defined(leanCompiler):
  95. setupJSgen(graph, module, idgen)
  96. else:
  97. nil
  98. of EvalPass, InterpreterPass:
  99. setupEvalGen(graph, module, idgen)
  100. of GenDependPass:
  101. setupDependPass(graph, module, idgen)
  102. of Docgen2Pass:
  103. when not defined(leanCompiler):
  104. openHtml(graph, module, idgen)
  105. else:
  106. nil
  107. of Docgen2TexPass:
  108. when not defined(leanCompiler):
  109. openTex(graph, module, idgen)
  110. else:
  111. nil
  112. of Docgen2JsonPass:
  113. when not defined(leanCompiler):
  114. openJson(graph, module, idgen)
  115. else:
  116. nil
  117. of SemPass:
  118. nil
  119. of NonePass:
  120. doAssert false, "use setPipeLinePass to set a proper PipelinePass"
  121. nil
  122. if stream == nil:
  123. let filename = toFullPathConsiderDirty(graph.config, fileIdx)
  124. s = llStreamOpen(filename, fmRead)
  125. if s == nil:
  126. rawMessage(graph.config, errCannotOpenFile, filename.string)
  127. return false
  128. else:
  129. s = stream
  130. while true:
  131. syntaxes.openParser(p, fileIdx, s, graph.cache, graph.config)
  132. if not belongsToStdlib(graph, module) or (belongsToStdlib(graph, module) and module.name.s == "distros"):
  133. # XXX what about caching? no processing then? what if I change the
  134. # modules to include between compilation runs? we'd need to track that
  135. # in ROD files. I think we should enable this feature only
  136. # for the interactive mode.
  137. if module.name.s != "nimscriptapi":
  138. processImplicitImports graph, graph.config.implicitImports, nkImportStmt, module, ctx, bModule, idgen
  139. processImplicitImports graph, graph.config.implicitIncludes, nkIncludeStmt, module, ctx, bModule, idgen
  140. checkFirstLineIndentation(p)
  141. block processCode:
  142. if graph.stopCompile(): break processCode
  143. var n = parseTopLevelStmt(p)
  144. if n.kind == nkEmpty: break processCode
  145. # read everything, no streaming possible
  146. var sl = newNodeI(nkStmtList, n.info)
  147. sl.add n
  148. while true:
  149. var n = parseTopLevelStmt(p)
  150. if n.kind == nkEmpty: break
  151. sl.add n
  152. prePass(ctx, sl)
  153. if sfReorder in module.flags or codeReordering in graph.config.features:
  154. sl = reorder(graph, sl, module)
  155. if graph.pipelinePass != EvalPass:
  156. message(graph.config, sl.info, hintProcessingStmt, $idgen[])
  157. var semNode = semWithPContext(ctx, sl)
  158. discard processPipeline(graph, semNode, bModule)
  159. closeParser(p)
  160. if s.kind != llsStdIn: break
  161. let finalNode = closePContext(graph, ctx, nil)
  162. case graph.pipelinePass
  163. of CgenPass:
  164. if bModule != nil:
  165. let disps = finalCodegenActions(graph, BModule(bModule), finalNode)
  166. if disps != nil:
  167. let ctx = preparePContext(graph, module, idgen)
  168. for disp in disps:
  169. let retTyp = disp.sym.typ[0]
  170. if retTyp != nil:
  171. # todo properly semcheck the code of dispatcher?
  172. createTypeBoundOps(graph, ctx, retTyp, disp.info, idgen)
  173. genProcAux(BModule(bModule), disp.sym)
  174. discard closePContext(graph, ctx, nil)
  175. of JSgenPass:
  176. when not defined(leanCompiler):
  177. discard finalJSCodeGen(graph, bModule, finalNode)
  178. of EvalPass, InterpreterPass:
  179. discard interpreterCode(bModule, finalNode)
  180. of SemPass, GenDependPass:
  181. discard
  182. of Docgen2Pass, Docgen2TexPass:
  183. when not defined(leanCompiler):
  184. discard closeDoc(graph, bModule, finalNode)
  185. of Docgen2JsonPass:
  186. when not defined(leanCompiler):
  187. discard closeJson(graph, bModule, finalNode)
  188. of NonePass:
  189. doAssert false, "use setPipeLinePass to set a proper PipelinePass"
  190. if graph.config.backend notin {backendC, backendCpp, backendObjc}:
  191. # We only write rod files here if no C-like backend is active.
  192. # The C-like backends have been patched to support the IC mechanism.
  193. # They are responsible for closing the rod files. See `cbackend.nim`.
  194. closeRodFile(graph, module)
  195. result = true
  196. proc compilePipelineModule*(graph: ModuleGraph; fileIdx: FileIndex; flags: TSymFlags, fromModule: PSym = nil): PSym =
  197. var flags = flags
  198. if fileIdx == graph.config.projectMainIdx2: flags.incl sfMainModule
  199. result = graph.getModule(fileIdx)
  200. template processModuleAux(moduleStatus) =
  201. onProcessing(graph, fileIdx, moduleStatus, fromModule = fromModule)
  202. var s: PLLStream
  203. if sfMainModule in flags:
  204. if graph.config.projectIsStdin: s = stdin.llStreamOpen
  205. elif graph.config.projectIsCmd: s = llStreamOpen(graph.config.cmdInput)
  206. discard processPipelineModule(graph, result, idGeneratorFromModule(result), s)
  207. if result == nil:
  208. var cachedModules: seq[FileIndex]
  209. result = moduleFromRodFile(graph, fileIdx, cachedModules)
  210. let path = toFullPath(graph.config, fileIdx)
  211. let filename = AbsoluteFile path
  212. if fileExists(filename): # it could be a stdinfile
  213. graph.cachedFiles[path] = $secureHashFile(path)
  214. if result == nil:
  215. result = newModule(graph, fileIdx)
  216. result.flags.incl flags
  217. registerModule(graph, result)
  218. processModuleAux("import")
  219. else:
  220. if sfSystemModule in flags:
  221. graph.systemModule = result
  222. partialInitModule(result, graph, fileIdx, filename)
  223. for m in cachedModules:
  224. registerModuleById(graph, m)
  225. replayStateChanges(graph.packed[m.int].module, graph)
  226. replayGenericCacheInformation(graph, m.int)
  227. elif graph.isDirty(result):
  228. result.flags.excl sfDirty
  229. # reset module fields:
  230. initStrTables(graph, result)
  231. result.ast = nil
  232. processModuleAux("import(dirty)")
  233. graph.markClientsDirty(fileIdx)
  234. proc importPipelineModule(graph: ModuleGraph; s: PSym, fileIdx: FileIndex): PSym =
  235. # this is called by the semantic checking phase
  236. assert graph.config != nil
  237. result = compilePipelineModule(graph, fileIdx, {}, s)
  238. graph.addDep(s, fileIdx)
  239. # keep track of import relationships
  240. if graph.config.hcrOn:
  241. graph.importDeps.mgetOrPut(FileIndex(s.position), @[]).add(fileIdx)
  242. #if sfSystemModule in result.flags:
  243. # localError(result.info, errAttemptToRedefine, result.name.s)
  244. # restore the notes for outer module:
  245. graph.config.notes =
  246. if graph.config.belongsToProjectPackage(s) or isDefined(graph.config, "booting"): graph.config.mainPackageNotes
  247. else: graph.config.foreignPackageNotes
  248. proc connectPipelineCallbacks*(graph: ModuleGraph) =
  249. graph.includeFileCallback = modules.includeModule
  250. graph.importModuleCallback = importPipelineModule
  251. proc compilePipelineSystemModule*(graph: ModuleGraph) =
  252. if graph.systemModule == nil:
  253. connectPipelineCallbacks(graph)
  254. graph.config.m.systemFileIdx = fileInfoIdx(graph.config,
  255. graph.config.libpath / RelativeFile"system.nim")
  256. discard graph.compilePipelineModule(graph.config.m.systemFileIdx, {sfSystemModule})
  257. proc compilePipelineProject*(graph: ModuleGraph; projectFileIdx = InvalidFileIdx) =
  258. connectPipelineCallbacks(graph)
  259. let conf = graph.config
  260. wantMainModule(conf)
  261. configComplete(graph)
  262. let systemFileIdx = fileInfoIdx(conf, conf.libpath / RelativeFile"system.nim")
  263. let projectFile = if projectFileIdx == InvalidFileIdx: conf.projectMainIdx else: projectFileIdx
  264. conf.projectMainIdx2 = projectFile
  265. let packSym = getPackage(graph, projectFile)
  266. graph.config.mainPackageId = packSym.getPackageId
  267. graph.importStack.add projectFile
  268. if projectFile == systemFileIdx:
  269. discard graph.compilePipelineModule(projectFile, {sfMainModule, sfSystemModule})
  270. else:
  271. graph.compilePipelineSystemModule()
  272. discard graph.compilePipelineModule(projectFile, {sfMainModule})