summary refs log tree commit diff stats
path: root/compiler/depends.nim
diff options
context:
space:
mode:
Diffstat (limited to 'compiler/depends.nim')
-rw-r--r--compiler/depends.nim110
1 files changed, 110 insertions, 0 deletions
diff --git a/compiler/depends.nim b/compiler/depends.nim
new file mode 100644
index 000000000..638f1eb51
--- /dev/null
+++ b/compiler/depends.nim
@@ -0,0 +1,110 @@
+#
+#
+#           The Nim Compiler
+#        (c) Copyright 2012 Andreas Rumpf
+#
+#    See the file "copying.txt", included in this
+#    distribution, for details about the copyright.
+#
+
+# This module implements a dependency file generator.
+
+import options, ast, ropes, pathutils, msgs, lineinfos
+
+import modulegraphs
+
+import std/[os, parseutils]
+import std/strutils except addf
+import std/private/globs
+
+when defined(nimPreviewSlimSystem):
+  import std/assertions
+
+
+type
+  TGen = object of PPassContext
+    module: PSym
+    config: ConfigRef
+    graph: ModuleGraph
+  PGen = ref TGen
+
+  Backend = ref object of RootRef
+    dotGraph: Rope
+
+proc addDependencyAux(b: Backend; importing, imported: string) =
+  b.dotGraph.addf("\"$1\" -> \"$2\";$n", [rope(importing), rope(imported)])
+  # s1 -> s2_4[label="[0-9]"];
+
+proc toNimblePath(s: string, isStdlib: bool): string =
+  const stdPrefix = "std/"
+  const pkgPrefix = "pkg/"
+  if isStdlib:
+    let sub = "lib/"
+    var start = s.find(sub)
+    if start < 0:
+      raiseAssert "unreachable"
+    else:
+      start += sub.len
+      let base = s[start..^1]
+
+      if base.startsWith("system") or base.startsWith("std"):
+        result = base
+      else:
+        for dir in stdlibDirs:
+          if base.startsWith(dir):
+            return stdPrefix & base.splitFile.name
+
+        result = stdPrefix & base
+  else:
+    var sub = getEnv("NIMBLE_DIR")
+    if sub.len == 0:
+      sub = ".nimble/pkgs/"
+    else:
+      sub.add "/pkgs/"
+    var start = s.find(sub)
+    if start < 0:
+      sub[^1] = '2'
+      sub.add '/'
+      start = s.find(sub) # /pkgs2
+      if start < 0:
+        return s
+
+    start += sub.len
+    start += skipUntil(s, '/', start)
+    start += 1
+    result = pkgPrefix & s[start..^1]
+
+proc addDependency(c: PPassContext, g: PGen, b: Backend, n: PNode) =
+  doAssert n.kind == nkSym, $n.kind
+
+  let path = splitFile(toProjPath(g.config, n.sym.position.FileIndex))
+  let modulePath = splitFile(toProjPath(g.config, g.module.position.FileIndex))
+  let parent = nativeToUnixPath(modulePath.dir / modulePath.name).toNimblePath(belongsToStdlib(g.graph, g.module))
+  let child = nativeToUnixPath(path.dir / path.name).toNimblePath(belongsToStdlib(g.graph, n.sym))
+  addDependencyAux(b, parent, child)
+
+proc addDotDependency*(c: PPassContext, n: PNode): PNode =
+  result = n
+  let g = PGen(c)
+  let b = Backend(g.graph.backend)
+  case n.kind
+  of nkImportStmt:
+    for i in 0..<n.len:
+      addDependency(c, g, b, n[i])
+  of nkFromStmt, nkImportExceptStmt:
+    addDependency(c, g, b, n[0])
+  of nkStmtList, nkBlockStmt, nkStmtListExpr, nkBlockExpr:
+    for i in 0..<n.len: discard addDotDependency(c, n[i])
+  else:
+    discard
+
+proc generateDot*(graph: ModuleGraph; project: AbsoluteFile) =
+  let b = Backend(graph.backend)
+  discard writeRope("digraph $1 {$n$2}$n" % [
+      rope(project.splitFile.name), b.dotGraph],
+            changeFileExt(project, "dot"))
+
+proc setupDependPass*(graph: ModuleGraph; module: PSym; idgen: IdGenerator): PPassContext =
+  result = PGen(module: module, config: graph.config, graph: graph)
+  if graph.backend == nil:
+    graph.backend = Backend(dotGraph: "")