Last active
September 28, 2024 10:05
-
-
Save felko/472010a417f5c338c40cb5daae06d626 to your computer and use it in GitHub Desktop.
Trees that grow in Scala 3 using match types
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package amyc.ast | |
import amyc.utils.Positioned | |
/* A polymorphic module containing definitions of Amy trees. | |
* | |
* This trait represents either nominal trees (where names have not been resolved) | |
* or symbolic trees (where names/qualified names) have been resolved to unique identifiers. | |
* This is done by having two type fields within the module, | |
* which will be instantiated differently by the two different modules. | |
* | |
*/ | |
trait TreeModule { self => | |
/* Represents the type for the name for this tree module. | |
* (It will be either a plain string, or a unique symbol) | |
*/ | |
type Name | |
// Represents a name within an module | |
type QualifiedName | |
// A printer that knows how to print trees in this module. | |
// The modules will instantiate it as appropriate | |
val printer: Printer { val treeModule: self.type } | |
// Common ancestor for all trees | |
trait Tree extends Positioned { | |
override def toString: String = printer(this) | |
} | |
// Expressions | |
trait Expr extends Tree | |
type Ann[T] | |
// Variables | |
case class Variable(ann: Ann[Variable], name: Name) extends Expr | |
// Literals | |
trait Literal[+T] extends Expr { val value: T } | |
case class IntLiteral(ann: Ann[IntLiteral], value: Int) extends Literal[Int] | |
case class BooleanLiteral(ann: Ann[BooleanLiteral], value: Boolean) extends Literal[Boolean] | |
case class StringLiteral(ann: Ann[StringLiteral], value: String) extends Literal[String] | |
case class UnitLiteral(ann: Ann[UnitLiteral]) extends Literal[Unit] { val value: Unit = () } | |
// Binary operators | |
case class Plus(ann: Ann[Plus], lhs: Expr, rhs: Expr) extends Expr | |
case class Minus(ann: Ann[Minus], lhs: Expr, rhs: Expr) extends Expr | |
case class Times(ann: Ann[Times], lhs: Expr, rhs: Expr) extends Expr | |
case class Div(ann: Ann[Div], lhs: Expr, rhs: Expr) extends Expr | |
case class Mod(ann: Ann[Mod], lhs: Expr, rhs: Expr) extends Expr | |
case class LessThan(ann: Ann[LessThan], lhs: Expr, rhs: Expr) extends Expr | |
case class LessEquals(ann: Ann[LessEquals], lhs: Expr, rhs: Expr) extends Expr | |
case class And(ann: Ann[And], lhs: Expr, rhs: Expr) extends Expr | |
case class Or(ann: Ann[Or], lhs: Expr, rhs: Expr) extends Expr | |
case class Equals(ann: Ann[Equals], lhs: Expr, rhs: Expr) extends Expr | |
case class Concat(ann: Ann[Concat], lhs: Expr, rhs: Expr) extends Expr | |
// Unary operators | |
case class Not(ann: Ann[Not], e: Expr) extends Expr | |
case class Neg(ann: Ann[Neg], e: Expr) extends Expr | |
// Function/constructor call | |
case class Call(ann: Ann[Call], qname: QualifiedName, args: List[Expr]) extends Expr | |
// Constructor call (resolved after name analysis) | |
case class Constr(ann: Ann[Constr], qname: QualifiedName, args: List[Expr]) extends Expr | |
// The ; operator | |
case class Sequence(ann: Ann[Sequence], e1: Expr, e2: Expr) extends Expr | |
// Local variable definition | |
case class Let(ann: Ann[Let], df: ParamDef, value: Expr, body: Expr) extends Expr | |
// If-then-else | |
case class Ite(ann: Ann[Ite], cond: Expr, thenn: Expr, elze: Expr) extends Expr | |
// Pattern matching | |
case class Match(ann: Ann[Match], scrut: Expr, cases: List[MatchCase]) extends Expr { | |
require(cases.nonEmpty) | |
} | |
// Represents a computational error; prints its message, then exits | |
case class Error(ann: Ann[Error], msg: Expr) extends Expr | |
// Cases and patterns for Match expressions | |
case class MatchCase(pat: Pattern, expr: Expr) extends Tree | |
abstract class Pattern extends Tree | |
case class WildcardPattern() extends Pattern // _ | |
case class IdPattern(name: Name) extends Pattern // x | |
case class LiteralPattern[+T](lit: Literal[T]) extends Pattern // 42, true | |
case class CaseClassPattern(constr: QualifiedName, args: List[Pattern]) extends Pattern // C(arg1, arg2) | |
// Annotations | |
case class Annotation(name: String) extends Tree | |
// Definitions | |
trait Definition extends Tree { val name: Name } | |
case class ModuleDef(name: Name, defs: List[ClassOrFunDef], optExpr: Option[Expr]) extends Definition | |
trait ClassOrFunDef extends Definition | |
case class AbstractClassDef(name: Name) extends ClassOrFunDef | |
case class CaseClassDef(name: Name, fields: List[TypeTree], parent: Name) extends ClassOrFunDef | |
case class FunDef(annotations: List[Annotation], name: Name, params: List[ParamDef], retType: TypeTree, body: Expr) extends ClassOrFunDef { | |
def paramNames = params.map(_.name) | |
val hasTailRecAnnotation: Boolean = annotations contains Annotation("tailrec") | |
val hasTailRecModConsAnnotation: Boolean = annotations contains Annotation("tailrecModCons") | |
} | |
case class ParamDef(name: Name, tt: TypeTree) extends Definition | |
// Types | |
trait Type | |
case object IntType extends Type { | |
override def toString: String = "Int" | |
} | |
case object BooleanType extends Type { | |
override def toString: String = "Boolean" | |
} | |
case object StringType extends Type { | |
override def toString: String = "String" | |
} | |
case object UnitType extends Type { | |
override def toString: String = "Unit" | |
} | |
case class ClassType(qname: QualifiedName) extends Type { | |
override def toString: String = printer.printQName(qname)(false).print | |
} | |
// A wrapper for types that is also a Tree (i.e. has a position) | |
case class TypeTree(tpe: Type) extends Tree | |
// All is wrapped in a program | |
case class Program(modules: List[ModuleDef]) extends Tree | |
} | |
/* A module containing trees where the names have not been resolved. | |
* Instantiates Name to String and QualifiedName to a pair of Strings | |
* representing (module, name) (where module is optional) | |
*/ | |
object NominalTreeModule extends TreeModule { | |
type Name = String | |
case class QualifiedName(module: Option[String], name: String) { | |
override def toString: String = printer.printQName(this)(false).print | |
} | |
type Ann[T] = T match { | |
case Constr => Nothing | |
case _ => Unit | |
} | |
val printer = NominalPrinter | |
} | |
/* A module containing trees where the names have been resolved to unique identifiers. | |
* Both Name and ModuleName are instantiated to Identifier. | |
*/ | |
object SymbolicTreeModule extends TreeModule { | |
type Name = Identifier | |
type QualifiedName = Identifier | |
type Ann[T] = T match { | |
case _ => Unit | |
} | |
val printer = SymbolicPrinter | |
} | |
object TCOTreeModule extends TreeModule { | |
type Name = Identifier | |
type QualifiedName = Identifier | |
enum TailCallFlag: | |
case TailCall | |
case NonTailCall | |
enum TailCallModConsFlag: | |
case NotTailModCons | |
case TailModConsAtArg(argIndex: Int) | |
type Ann[T] = T match { | |
case Call => TailCallFlag | |
case Constr => TailCallModConsFlag | |
case _ => Unit | |
} | |
val printer = TCOPrinter | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
case class BooleanLiteral(ann: Ann[IntLiteral], value: Boolean) extends Literal[Boolean]
looks like a copypasto