Skip to content

Commit

Permalink
Sync IR changes (#225)
Browse files Browse the repository at this point in the history
  • Loading branch information
waterlens authored Oct 3, 2024
1 parent aa182b1 commit c389926
Show file tree
Hide file tree
Showing 38 changed files with 6,780 additions and 4,580 deletions.
23 changes: 23 additions & 0 deletions .github/workflows/nix.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
name: Cpp Backend CI with Nix

on:
pull_request:
push:
branches: [ mlscript ]

jobs:
build:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- name: Install Nix
uses: DeterminateSystems/nix-installer-action@main
- uses: rrbutani/use-nix-shell-action@v1
with:
devShell: .#default
- name: Install TypeScript
run: npm ci
- name: Run test
run: sbt -J-Xmx4096M -J-Xss4M test
- name: Check no changes
run: git diff-files -p --exit-code
27 changes: 0 additions & 27 deletions .github/workflows/scala.yml

This file was deleted.

6 changes: 3 additions & 3 deletions build.sbt
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import Wart._

enablePlugins(ScalaJSPlugin)

ThisBuild / scalaVersion := "2.13.12"
ThisBuild / scalaVersion := "2.13.14"
ThisBuild / version := "0.1.0-SNAPSHOT"
ThisBuild / organization := "io.lptk"
ThisBuild / organizationName := "LPTK"
Expand Down Expand Up @@ -52,7 +52,7 @@ lazy val mlscript = crossProject(JSPlatform, JVMPlatform).in(file("."))
)
.jsSettings(
scalaJSUseMainModuleInitializer := true,
libraryDependencies += "org.scala-js" %%% "scalajs-dom" % "2.1.0",
libraryDependencies += "org.scala-js" %%% "scalajs-dom" % "2.2.0",
)

lazy val mlscriptJVM = mlscript.jvm
Expand All @@ -79,7 +79,7 @@ lazy val ts2mlsTest = project.in(file("ts2mls"))
lazy val compiler = crossProject(JSPlatform, JVMPlatform).in(file("compiler"))
.settings(
name := "mlscript-compiler",
scalaVersion := "3.1.3",
scalaVersion := "3.3.3",
sourceDirectory := baseDirectory.value.getParentFile()/"shared",
watchSources += WatchSource(
baseDirectory.value.getParentFile()/"shared"/"test"/"diff", "*.mls", NothingFilter),
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
package mlscript.compiler.optimizer
package mlscript.compiler.utils

enum Document:
case Indented(content: Document)
Expand All @@ -10,7 +10,7 @@ enum Document:
def <:>(other: Document) = line(List(this, other))
def <#>(other: Document) = line(List(this, other), sep = "")

override def toString(): String = print
override def toString: String = print

def print: String = {
val sb = StringBuffer()
Expand Down
210 changes: 210 additions & 0 deletions compiler/shared/main/scala/mlscript/compiler/codegen/CppAst.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,210 @@
package mlscript.compiler.codegen.cpp

import mlscript._
import mlscript.utils._
import mlscript.utils.shorthands._
import mlscript.compiler.utils._

import scala.language.implicitConversions

given Conversion[String, Document] = raw

enum Specifier:
case Extern
case Static
case Inline

def toDocument = raw:
this match
case Extern => "extern"
case Static => "static"
case Inline => "inline"

override def toString: Str = toDocument.print

object Type:
def toDocuments(args: Ls[Type], sep: Document, extraTypename: Bool = false): Document =
args.iterator.zipWithIndex.map {
case (x, 0) =>
x.toDocument(extraTypename)
case (x, _) =>
sep <#> x.toDocument(extraTypename)
}.fold(raw(""))(_ <#> _)

def toDocuments(args: Ls[(Str, Type)], sep: Document): Document =
args.iterator.zipWithIndex.map {
case (x, 0) =>
x._2.toDocument() <:> raw(x._1)
case (x, _) =>
sep <#> x._2.toDocument() <:> raw(x._1)
}.fold(raw(""))(_ <#> _)

enum Type:
case Prim(name: Str)
case Ptr(inner: Type)
case Ref(inner: Type)
case Array(inner: Type, size: Opt[Int])
case FuncPtr(ret: Type, args: List[Type])
case Struct(name: Str)
case Enum(name: Str)
case Template(name: Str, args: List[Type])
case Var(name: Str)
case Qualifier(inner: Type, qual: Str)

def toDocument(extraTypename: Bool = false): Document =
def aux(x: Type): Document = x match
case Prim(name) => name
case Ptr(inner) => aux(inner) <#> "*"
case Ref(inner) => aux(inner) <#> "&"
case Array(inner, size) => aux(inner) <#> "[" <#> size.fold(raw(""))(x => x.toString) <#> "]"
case FuncPtr(ret, args) => aux(ret) <#> "(" <#> Type.toDocuments(args, sep = ", ") <#> ")"
case Struct(name) => s"struct $name"
case Enum(name) => s"enum $name"
case Template(name, args) => s"$name" <#> "<" <#> Type.toDocuments(args, sep = ", ") <#> ">"
case Var(name) => name
case Qualifier(inner, qual) => aux(inner) <:> qual
aux(this)

override def toString: Str = toDocument().print

object Stmt:
def toDocuments(decl: Ls[Decl], stmts: Ls[Stmt]): Document =
stack_list(decl.map(_.toDocument) ++ stmts.map(_.toDocument))

enum Stmt:
case AutoBind(lhs: Ls[Str], rhs: Expr)
case Assign(lhs: Str, rhs: Expr)
case Return(expr: Expr)
case If(cond: Expr, thenStmt: Stmt, elseStmt: Opt[Stmt])
case While(cond: Expr, body: Stmt)
case For(init: Stmt, cond: Expr, update: Stmt, body: Stmt)
case ExprStmt(expr: Expr)
case Break
case Continue
case Block(decl: Ls[Decl], stmts: Ls[Stmt])
case Switch(expr: Expr, cases: Ls[(Expr, Stmt)])
case Raw(stmt: Str)

def toDocument: Document =
def aux(x: Stmt): Document = x match
case AutoBind(lhs, rhs) =>
lhs match
case Nil => rhs.toDocument
case x :: Nil => "auto" <:> x <:> "=" <:> rhs.toDocument <#> ";"
case _ => "auto" <:> lhs.mkString("[", ",", "]") <:> "=" <:> rhs.toDocument <#> ";"
case Assign(lhs, rhs) => lhs <#> " = " <#> rhs.toDocument <#> ";"
case Return(expr) => "return " <#> expr.toDocument <#> ";"
case If(cond, thenStmt, elseStmt) =>
"if (" <#> cond.toDocument <#> ")" <#> thenStmt.toDocument <:> elseStmt.fold(raw(""))(x => "else" <:> x.toDocument)
case While(cond, body) =>
"while (" <#> cond.toDocument <#> ")" <#> body.toDocument
case For(init, cond, update, body) =>
"for (" <#> init.toDocument <#> "; " <#> cond.toDocument <#> "; " <#> update.toDocument <#> ")" <#> body.toDocument
case ExprStmt(expr) => expr.toDocument <#> ";"
case Break => "break;"
case Continue => "continue;"
case Block(decl, stmts) =>
stack(
"{",
Stmt.toDocuments(decl, stmts) |> indent,
"}")
case Switch(expr, cases) =>
"switch (" <#> expr.toDocument <#> ")" <#> "{" <#> stack_list(cases.map {
case (cond, stmt) => "case " <#> cond.toDocument <#> ":" <#> stmt.toDocument
}) <#> "}"
case Raw(stmt) => stmt
aux(this)

object Expr:
def toDocuments(args: Ls[Expr], sep: Document): Document =
args.zipWithIndex.map {
case (x, i) =>
if i == 0 then x.toDocument
else sep <#> x.toDocument
}.fold(raw(""))(_ <#> _)

enum Expr:
case Var(name: Str)
case IntLit(value: BigInt)
case DoubleLit(value: Double)
case StrLit(value: Str)
case CharLit(value: Char)
case Call(func: Expr, args: Ls[Expr])
case Member(expr: Expr, member: Str)
case Index(expr: Expr, index: Expr)
case Unary(op: Str, expr: Expr)
case Binary(op: Str, lhs: Expr, rhs: Expr)
case Initializer(exprs: Ls[Expr])
case Constructor(name: Str, init: Expr)

def toDocument: Document =
def aux(x: Expr): Document = x match
case Var(name) => name
case IntLit(value) => value.toString
case DoubleLit(value) => value.toString
case StrLit(value) => s"\"$value\"" // need more reliable escape utils
case CharLit(value) => value.toInt.toString
case Call(func, args) => aux(func) <#> "(" <#> Expr.toDocuments(args, sep = ", ") <#> ")"
case Member(expr, member) => aux(expr) <#> "->" <#> member
case Index(expr, index) => aux(expr) <#> "[" <#> aux(index) <#> "]"
case Unary(op, expr) => "(" <#> op <#> aux(expr) <#> ")"
case Binary(op, lhs, rhs) => "(" <#> aux(lhs) <#> op <#> aux(rhs) <#> ")"
case Initializer(exprs) => "{" <#> Expr.toDocuments(exprs, sep = ", ") <#> "}"
case Constructor(name, init) => name <#> init.toDocument
aux(this)

case class CompilationUnit(includes: Ls[Str], decls: Ls[Decl], defs: Ls[Def]):
def toDocument: Document =
stack_list(includes.map(x => raw(x)) ++ decls.map(_.toDocument) ++ defs.map(_.toDocument))
def toDocumentWithoutHidden: Document =
val hiddenNames = Set(
"HiddenTheseEntities", "True", "False", "Callable", "List", "Cons", "Nil", "Option", "Some", "None", "Pair", "Tuple2", "Tuple3", "Nat", "S", "O"
)
stack_list(defs.filterNot {
case Def.StructDef(name, _, _, _) => hiddenNames.contains(name.stripPrefix("_mls_"))
case _ => false
}.map(_.toDocument))

enum Decl:
case StructDecl(name: Str)
case EnumDecl(name: Str)
case FuncDecl(ret: Type, name: Str, args: Ls[Type])
case VarDecl(name: Str, typ: Type)

def toDocument: Document =
def aux(x: Decl): Document = x match
case StructDecl(name) => s"struct $name;"
case EnumDecl(name) => s"enum $name;"
case FuncDecl(ret, name, args) => ret.toDocument() <#> s" $name(" <#> Type.toDocuments(args, sep = ", ") <#> ");"
case VarDecl(name, typ) => typ.toDocument() <#> s" $name;"
aux(this)

enum Def:
case StructDef(name: Str, fields: Ls[(Str, Type)], inherit: Opt[Ls[Str]], methods: Ls[Def] = Ls.empty)
case EnumDef(name: Str, fields: Ls[(Str, Opt[Int])])
case FuncDef(specret: Type, name: Str, args: Ls[(Str, Type)], body: Stmt.Block, or: Bool = false, virt: Bool = false)
case VarDef(typ: Type, name: Str, init: Opt[Expr])
case RawDef(raw: Str)

def toDocument: Document =
def aux(x: Def): Document = x match
case StructDef(name, fields, inherit, defs) =>
stack(
s"struct $name" <#> (if inherit.nonEmpty then ": public" <:> inherit.get.mkString(", ") else "" ) <:> "{",
stack_list(fields.map {
case (name, typ) => typ.toDocument() <#> " " <#> name <#> ";"
}) |> indent,
stack_list(defs.map(_.toDocument)) |> indent,
"};"
)
case EnumDef(name, fields) =>
s"enum $name" <:> "{" <#> stack_list(fields.map {
case (name, value) => value.fold(s"$name")(x => s"$name = $x")
}) <#> "};"
case FuncDef(specret, name, args, body, or, virt) =>
(if virt then "virtual " else "")
<#> specret.toDocument() <#> s" $name(" <#> Type.toDocuments(args, sep = ", ") <#> ")" <#> (if or then " override" else "") <#> body.toDocument
case VarDef(typ, name, init) =>
typ.toDocument() <#> s" $name" <#> init.fold(raw(""))(x => " = " <#> x.toDocument) <#> raw(";")
case RawDef(x) => x
aux(this)
Loading

0 comments on commit c389926

Please sign in to comment.