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