Skip to content

Commit

Permalink
Add methods to remove unused values (treeshake)
Browse files Browse the repository at this point in the history
  • Loading branch information
johnynek committed Nov 22, 2024
1 parent be453d5 commit 78560b3
Show file tree
Hide file tree
Showing 2 changed files with 32 additions and 5 deletions.
16 changes: 12 additions & 4 deletions core/src/main/scala/org/bykn/bosatsu/Package.scala
Original file line number Diff line number Diff line change
Expand Up @@ -64,16 +64,17 @@ object Package {
type Parsed = Package[PackageName, Unit, Unit, List[Statement]]
type Resolved =
FixPackage[Unit, Unit, (List[Statement], ImportMap[PackageName, Unit])]
type TypedProgram[T] = (
Program[TypeEnv[Kind.Arg], TypedExpr[T], Any],
ImportMap[Interface, NonEmptyList[Referant[Kind.Arg]]]
)
type Typed[T] = Package[
Interface,
NonEmptyList[Referant[Kind.Arg]],
Referant[
Kind.Arg
],
(
Program[TypeEnv[Kind.Arg], TypedExpr[T], Any],
ImportMap[Interface, NonEmptyList[Referant[Kind.Arg]]]
)
TypedProgram[T]
]
type Inferred = Typed[Declaration]

Expand Down Expand Up @@ -528,6 +529,13 @@ object Package {
(Package.Interface, ImportedName[NonEmptyList[Referant[Kind.Arg]]])
] =
pack.program._2(n)

def filterLets(fn: Identifier => Boolean): Typed[A] = {
val lets = pack.program._1.lets
val prog1: Program[TypeEnv[Kind.Arg], TypedExpr[A], Any] =
pack.program._1.copy(lets = lets.filter { case (b, _, _) => fn(b) })
pack.copy(program = (prog1, pack.program._2))
}
}

def orderByName[A, B, C, D]: Order[Package[A, B, C, D]] =
Expand Down
21 changes: 20 additions & 1 deletion core/src/main/scala/org/bykn/bosatsu/PackageMap.scala
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
package org.bykn.bosatsu

import org.bykn.bosatsu.graph.Memoize
import org.bykn.bosatsu.graph.{Dag, Memoize}
import cats.{Foldable, Monad, Show}
import cats.data.{
Ior,
Expand Down Expand Up @@ -90,6 +90,25 @@ object PackageMap {
// convenience for type inference
def toAnyTyped[A](p: Typed[A]): Typed[Any] = p

def treeShake[A](p: Typed[A], roots: Set[(PackageName, Identifier)]): Typed[A] = {
type Ident = (PackageName, Identifier)
def dependency(a: Ident): Iterable[Ident] =
p.toMap.get(a._1) match {
case Some(pack) =>
pack.lets.flatMap { case (_, _, te) => te.globals }
case None => Nil
}

val keep = Dag.transitiveSet(roots.toList.sorted)(dependency)

val kept = p.toMap.iterator.map { case (_, pack) =>
pack.filterLets { nm => keep((pack.name, nm)) }
}
.toList

fromIterable(kept)
}

type Inferred = Typed[Declaration]

/** This builds a DAG of actual packages where names have been replaced by the
Expand Down

0 comments on commit 78560b3

Please sign in to comment.