From f913482589ae52728e81a83a32b7fd8d246067e5 Mon Sep 17 00:00:00 2001 From: phischu Date: Thu, 12 Dec 2024 12:44:37 +0100 Subject: [PATCH 01/31] Make sure unique name is actually fresh (#741) This fixes the failing program on effekt plots and fixes #732 --- .../src/main/scala/effekt/core/Renamer.scala | 2 +- .../scala/effekt/machine/Transformer.scala | 88 +++++++++++-------- 2 files changed, 50 insertions(+), 40 deletions(-) diff --git a/effekt/shared/src/main/scala/effekt/core/Renamer.scala b/effekt/shared/src/main/scala/effekt/core/Renamer.scala index b675b683e..f43b9c07b 100644 --- a/effekt/shared/src/main/scala/effekt/core/Renamer.scala +++ b/effekt/shared/src/main/scala/effekt/core/Renamer.scala @@ -22,7 +22,7 @@ class Renamer(names: Names = Names(Map.empty), prefix: String = "") extends core def freshIdFor(id: Id): Id = suffix = suffix + 1 - val uniqueName = if prefix.isEmpty then id.name.name + suffix.toString else prefix + suffix.toString + val uniqueName = if prefix.isEmpty then id.name.name + "_" + suffix.toString else prefix + suffix.toString names.idFor(uniqueName) def withBindings[R](ids: List[Id])(f: => R): R = diff --git a/effekt/shared/src/main/scala/effekt/machine/Transformer.scala b/effekt/shared/src/main/scala/effekt/machine/Transformer.scala index 86756b84c..34cd60ae5 100644 --- a/effekt/shared/src/main/scala/effekt/machine/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/machine/Transformer.scala @@ -111,10 +111,22 @@ object Transformer { noteDefinition(id, vparams.map(transform) ++ bparams.map(transform), freeParams.toList) - case Definition.Def(id, b @ core.New(impl)) => + case Definition.Def(id, block @ core.New(impl)) => // this is just a hack... - noteParameter(id, b.tpe) - case _ => () + noteParameter(id, block.tpe) + + case Definition.Def(id, core.BlockVar(alias, tpe, _)) => + getDefinition(alias) match { + case BlockInfo.Definition(free, params) => + noteDefinition(id, free, params) + } + + case Definition.Def(id, core.Unbox(_)) => + // TODO deal with this case + () + + case Definition.Let(_, _, _) => + () } @@ -129,12 +141,16 @@ object Transformer { } case (core.Definition.Def(id, core.BlockLit(tparams, cparams, vparams, bparams, body)), rest) => - Def(Label(transform(id), getBlocksParams(id)), transform(body), rest) + Def(transformLabel(id), transform(body), rest) case (core.Definition.Def(id, core.New(impl)), rest) => New(Variable(transform(id), transform(impl.interface)), transform(impl), rest) - case (d @ core.Definition.Def(_, _: core.BlockVar | _: core.Unbox), rest) => + case (core.Definition.Def(id, core.BlockVar(alias, tpe, _)), rest) => + Def(transformLabel(id), Jump(transformLabel(alias)), rest) + + case (d @ core.Definition.Def(_, _: core.Unbox), rest) => + // TODO deal with this case by substitution ErrorReporter.abort(s"block definition: $d") } @@ -303,21 +319,20 @@ object Transformer { } yield (values, blocks) def transformBlockArg(block: core.Block)(using BPC: BlocksParamsContext, DC: DeclarationContext, E: ErrorReporter): Binding[Variable] = block match { - case core.BlockVar(id, tpe, capt) if isDefinition(id) => - // Passing a top-level function directly, so we need to eta-expand turning it into a closure - // TODO cache the closure somehow to prevent it from being created on every call - val parameters = BPC.params(id) - val variable = Variable(freshName(id.name.name ++ "$closure"), Negative()) - val environment = getBlocksParams(id) - Binding { k => - New(variable, List(Clause(parameters, - // conceptually: Substitute(parameters zip parameters, Jump(...)) but the Substitute is a no-op here - Jump(Label(transform(id), environment)) - )), k(variable)) - } - - case core.BlockVar(id, tpe, capt) => - pure(Variable(transform(id), transform(tpe))) + case core.BlockVar(id, tpe, capt) => getBlockInfo(id) match { + case BlockInfo.Definition(_, parameters) => + // Passing a top-level function directly, so we need to eta-expand turning it into a closure + // TODO cache the closure somehow to prevent it from being created on every call + val variable = Variable(freshName(id.name.name ++ "$closure"), Negative()) + Binding { k => + New(variable, List(Clause(parameters, + // conceptually: Substitute(parameters zip parameters, Jump(...)) but the Substitute is a no-op here + Jump(transformLabel(id)) + )), k(variable)) + } + case BlockInfo.Parameter(tpe) => + pure(Variable(transform(id), transform(tpe))) + } case core.BlockLit(tparams, cparams, vparams, bparams, body) => noteParameters(bparams) @@ -489,6 +504,10 @@ object Transformer { case core.BlockType.Interface(symbol, targs) => Negative() } + def transformLabel(id: Id)(using BPC: BlocksParamsContext) = getDefinition(id) match { + case BlockInfo.Definition(freeParams, boundParams) => Label(transform(id), boundParams ++ freeParams) + } + def transform(id: Id): String = s"${id.name}_${id.id}" @@ -520,16 +539,9 @@ object Transformer { class BlocksParamsContext() { var info: Map[Symbol, BlockInfo] = Map.empty - var globals: Map[Id, Label] = Map.empty - - def definition(id: Id): BlockInfo.Definition = info(id) match { - case d : BlockInfo.Definition => d - case BlockInfo.Parameter(tpe) => sys error s"Expected a function definition, but got a block parameter: ${id}" - } - def params(id: Id): Environment = definition(id).params - def free(id: Id): Environment = definition(id).free } + enum BlockInfo { case Definition(free: Environment, params: Environment) case Parameter(tpe: core.BlockType) @@ -538,7 +550,7 @@ object Transformer { def DeclarationContext(using DC: DeclarationContext): DeclarationContext = DC def noteDefinition(id: Id, params: Environment, free: Environment)(using BC: BlocksParamsContext): Unit = - assert(!BC.info.isDefinedAt(id), s"Registering info twice for ${id} (was: ${BC.info(id)}, now: BlockInfo)") + assert(!BC.info.isDefinedAt(id), s"Registering info twice for ${id} (was: ${BC.info(id)}, now: Definition)") BC.info += (id -> BlockInfo.Definition(free, params)) def noteParameter(id: Id, tpe: core.BlockType)(using BC: BlocksParamsContext): Unit = @@ -550,18 +562,16 @@ object Transformer { case core.BlockParam(id, tpe, capt) => noteParameter(id, tpe) } - def noteGlobal(id: Id)(using BC: BlocksParamsContext): Unit = - BC.globals += (id -> Label(transform(id), Nil)) + def noteGlobal(id: Id)(using BPC: BlocksParamsContext): Unit = + BPC.globals += (id -> Label(transform(id), Nil)) - def getBlocksParams(id: Id)(using BC: BlocksParamsContext): Environment = BC.definition(id) match { - case BlockInfo.Definition(freeParams, blockParams) => blockParams ++ freeParams - } + def getBlockInfo(id: Id)(using BPC: BlocksParamsContext): BlockInfo = + BPC.info.getOrElse(id, sys error s"No block info for ${id}") - def isDefinition(id: Id)(using BC: BlocksParamsContext): Boolean = - BC.info(id) match { - case d: BlockInfo.Definition => true - case _ => false - } + def getDefinition(id: Id)(using BPC: BlocksParamsContext): BlockInfo.Definition = getBlockInfo(id) match { + case d : BlockInfo.Definition => d + case BlockInfo.Parameter(tpe) => sys error s"Expected a function getDefinition, but got a block parameter: ${id}" + } case class Binding[A](run: (A => Statement) => Statement) { def flatMap[B](rest: A => Binding[B]): Binding[B] = { From 94c88518cd6ee65859c7014a6d27a92f1bc5b99d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ji=C5=99=C3=AD=20Bene=C5=A1?= Date: Thu, 12 Dec 2024 17:34:53 +0100 Subject: [PATCH 02/31] Revert "Cherry-pick inlining fix from #725" (#745) Reverts effekt-lang/effekt#730 as it seems like an incorrect fix, judging both by the Effekt Working Group meeting, and the graphs after merging it: ![image](https://github.com/user-attachments/assets/67f28d10-ef57-4529-b47d-4df1d342bd63) ![image](https://github.com/user-attachments/assets/e6de9240-d9fd-4aeb-8f45-79a225307811) --- effekt/shared/src/main/scala/effekt/core/Inline.scala | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/effekt/shared/src/main/scala/effekt/core/Inline.scala b/effekt/shared/src/main/scala/effekt/core/Inline.scala index 2ee139d9b..18f91c607 100644 --- a/effekt/shared/src/main/scala/effekt/core/Inline.scala +++ b/effekt/shared/src/main/scala/effekt/core/Inline.scala @@ -98,7 +98,8 @@ object Inline { def blockDefFor(id: Id)(using ctx: InlineContext): Option[Block] = ctx.defs.get(id) map { - case Definition.Def(id, block) => rewrite(block) + // TODO rewriting here leads to a stack overflow in one test, why? + case Definition.Def(id, block) => block //rewrite(block) case Definition.Let(id, _, binding) => INTERNAL_ERROR("Should not happen") } From 6753ebcfabe79c2155eb6af0aef3e7e9a13614b0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ji=C5=99=C3=AD=20Bene=C5=A1?= Date: Fri, 13 Dec 2024 17:53:40 +0100 Subject: [PATCH 03/31] Add a regression check for #692 (#693) Adds a quick regression test for a handler in tail position that uses a mutable variable (minimised reproduction of the underlying issue). The test passes on current `master` after #692. Crashes the compiler on `v0.9.0`, but works on `v0.8.0`. --- examples/pos/mutabletailrec.check | 1 + examples/pos/mutabletailrec.effekt | 19 +++++++++++++++++++ 2 files changed, 20 insertions(+) create mode 100644 examples/pos/mutabletailrec.check create mode 100644 examples/pos/mutabletailrec.effekt diff --git a/examples/pos/mutabletailrec.check b/examples/pos/mutabletailrec.check new file mode 100644 index 000000000..d00491fd7 --- /dev/null +++ b/examples/pos/mutabletailrec.check @@ -0,0 +1 @@ +1 diff --git a/examples/pos/mutabletailrec.effekt b/examples/pos/mutabletailrec.effekt new file mode 100644 index 000000000..346c97563 --- /dev/null +++ b/examples/pos/mutabletailrec.effekt @@ -0,0 +1,19 @@ +module examples/pos/mutabletailrec + +effect tagging(tag: String): Unit + +def handle { prog: () => Unit / tagging }: Unit = { + try { + prog() + } with tagging { tag => + var total = 0 + total = total + 1 + println(total) + + resume(()) + } +} + +def main() = handle { + do tagging("foo") +} From eb662be391d19eda0194af32eb7362c49032bbb5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ji=C5=99=C3=AD=20Bene=C5=A1?= Date: Sun, 15 Dec 2024 18:56:36 +0100 Subject: [PATCH 04/31] Do not abort after successful compilation on js-web (#747) Resolves #717 I am fully aware that this is not the best way to do things, a larger refactor would be needed (see the TODO in code). However, it's a somewhat pressing issue as we have users trying to work with `js-web` blocked on this... As a bonus, I added a `id=app` div to the generated HTML. --- effekt/jvm/src/main/scala/effekt/Runner.scala | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/effekt/jvm/src/main/scala/effekt/Runner.scala b/effekt/jvm/src/main/scala/effekt/Runner.scala index ed582464c..ab26700e2 100644 --- a/effekt/jvm/src/main/scala/effekt/Runner.scala +++ b/effekt/jvm/src/main/scala/effekt/Runner.scala @@ -216,11 +216,16 @@ object JSWebRunner extends Runner[String] { | + |
| | |""".stripMargin IO.createFile(htmlFilePath, htmlContent, false) - C.abort(s"Open file://${htmlFilePath} in your browser or include ${jsFilePath}.") + + // TODO: In ErrorReporter, add a way to terminate the program with a message, but not report a exit failure. + // Workaround: print and then 'exit(0)' + println(s"Open file://${htmlFilePath} in your browser or include ${jsFilePath}.") + scala.sys.exit(0) } trait ChezRunner extends Runner[String] { From ff669930823f9311ce2e0df387f6c33a5c62d971 Mon Sep 17 00:00:00 2001 From: "effekt-updater[bot]" <181701480+effekt-updater[bot]@users.noreply.github.com> Date: Mon, 16 Dec 2024 03:45:08 +0000 Subject: [PATCH 05/31] Bump version to 0.14.0 --- package.json | 2 +- pom.xml | 2 +- project/EffektVersion.scala | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/package.json b/package.json index 54af68e15..33ed2f9dd 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "@effekt-lang/effekt", "author": "Jonathan BrachthΓ€user", - "version": "0.13.0", + "version": "0.14.0", "repository": { "type": "git", "url": "git+https://github.com/effekt-lang/effekt.git" diff --git a/pom.xml b/pom.xml index be08200da..5fccb3829 100644 --- a/pom.xml +++ b/pom.xml @@ -7,7 +7,7 @@ de.bstudios Effekt Effekt - 0.13.0 + 0.14.0 4.0.0 diff --git a/project/EffektVersion.scala b/project/EffektVersion.scala index 5b896a07d..721c68d61 100644 --- a/project/EffektVersion.scala +++ b/project/EffektVersion.scala @@ -1,4 +1,4 @@ // Don't change this file without changing the CI too! import sbt.* import sbt.Keys.* -object EffektVersion { lazy val effektVersion = "0.13.0" } +object EffektVersion { lazy val effektVersion = "0.14.0" } From a4418db63ff53325a8710a31ca9d9c3165a9552d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marcial=20Gai=C3=9Fert?= Date: Wed, 18 Dec 2024 17:03:39 +0100 Subject: [PATCH 06/31] Stdlib: Add simple libraries: Scanner, Json (#631) Based on #622. This adds: - Some (minimal) features in the effekt stdlib and the subset of #527 that it depends on - A simple Scanner library (lexing string literals and numbers for now) - A library to parse and print (minified, not pretty) json --------- Co-authored-by: Philipp Schuster --- .../test/scala/effekt/ChezSchemeTests.scala | 4 + .../jvm/src/test/scala/effekt/LLVMTests.scala | 2 + .../benchmarks/input_output/dyck_one.check | 1 + .../benchmarks/input_output/dyck_one.effekt | 56 +++ .../input_output/number_matrix.check | 1 + .../input_output/number_matrix.effekt | 90 ++++ .../input_output/word_count_ascii.effekt | 11 + .../input_output/word_count_utf8.effekt | 13 + examples/stdlib/stream/sum_of_squares.effekt | 1 + libraries/common/char.effekt | 70 ++-- libraries/common/effekt.effekt | 6 +- libraries/common/json.effekt | 385 ++++++++++++++++++ libraries/common/scanner.effekt | 171 ++++++++ libraries/common/stream.effekt | 295 ++++++++------ 14 files changed, 948 insertions(+), 158 deletions(-) create mode 100644 examples/benchmarks/input_output/dyck_one.check create mode 100644 examples/benchmarks/input_output/dyck_one.effekt create mode 100644 examples/benchmarks/input_output/number_matrix.check create mode 100644 examples/benchmarks/input_output/number_matrix.effekt create mode 100644 libraries/common/json.effekt create mode 100644 libraries/common/scanner.effekt diff --git a/effekt/jvm/src/test/scala/effekt/ChezSchemeTests.scala b/effekt/jvm/src/test/scala/effekt/ChezSchemeTests.scala index 00f2f659f..89a0b0eee 100644 --- a/effekt/jvm/src/test/scala/effekt/ChezSchemeTests.scala +++ b/effekt/jvm/src/test/scala/effekt/ChezSchemeTests.scala @@ -26,8 +26,12 @@ abstract class ChezSchemeTests extends EffektTests { examplesDir / "pos" / "bidirectional", examplesDir / "pos" / "object", examplesDir / "pos" / "type_omission_op.effekt", + + // filesystem operations and bytearrays are not yet supported in our Chez backend examplesDir / "benchmarks" / "input_output" / "word_count_ascii.effekt", examplesDir / "benchmarks" / "input_output" / "word_count_utf8.effekt", + examplesDir / "benchmarks" / "input_output" / "dyck_one.effekt", + examplesDir / "benchmarks" / "input_output" / "number_matrix.effekt", // unsafe continuations are not yet supported in our Chez backend examplesDir / "pos" / "unsafe_cont.effekt", diff --git a/effekt/jvm/src/test/scala/effekt/LLVMTests.scala b/effekt/jvm/src/test/scala/effekt/LLVMTests.scala index ee89dd933..6359f5f30 100644 --- a/effekt/jvm/src/test/scala/effekt/LLVMTests.scala +++ b/effekt/jvm/src/test/scala/effekt/LLVMTests.scala @@ -39,6 +39,8 @@ class LLVMTests extends EffektTests { examplesDir / "benchmarks" / "are_we_fast_yet" / "queens.effekt", examplesDir / "benchmarks" / "input_output" / "word_count_ascii.effekt", examplesDir / "benchmarks" / "input_output" / "word_count_utf8.effekt", + examplesDir / "benchmarks" / "input_output" / "dyck_one.effekt", + examplesDir / "benchmarks" / "input_output" / "number_matrix.effekt", ) /** diff --git a/examples/benchmarks/input_output/dyck_one.check b/examples/benchmarks/input_output/dyck_one.check new file mode 100644 index 000000000..4c009fb2f --- /dev/null +++ b/examples/benchmarks/input_output/dyck_one.check @@ -0,0 +1 @@ +206 \ No newline at end of file diff --git a/examples/benchmarks/input_output/dyck_one.effekt b/examples/benchmarks/input_output/dyck_one.effekt new file mode 100644 index 000000000..d34664af8 --- /dev/null +++ b/examples/benchmarks/input_output/dyck_one.effekt @@ -0,0 +1,56 @@ +import examples/benchmarks/runner + +import io/error +import io/filesystem +import stream +import scanner + +// dyck_one.txt +// ((((()())(()())(()()))((()())(()())(()()))((()())(()())(()()))((()())(()())(()())))... + +def emitTree(n: Int): Unit / emit[Byte] = + if (n <= 0) { + () + } else { + do emit(40.toByte) + repeat(n) { emitTree(n - 1) } + do emit(41.toByte) + } + +type Tree { + Leaf() + Node(children: List[Tree]) +} + +def readTree(): Tree / { Scan[Byte], stop } = { + readIf[Byte] { b => b.toInt == 40 } + val children = collectList[Tree] { many { readTree() } } + skipIf[Byte] { b => b.toInt == 41 } + Node(children) +} + +def size(tree: Tree): Int = + tree match { + case Leaf() => 1 + case Node(children) => 1 + sum { + for[Tree] { children.each } { tree => do emit(size(tree)) } + } + } + +def run(n: Int): Int = { + with on[IOError].panic + + val filename = "/tmp/dyck_one.txt" + + val _ = { + with writeFile(filename) + emitTree(n) + } + + with readFile(filename) + with returning::scanner[Byte, Int] + attempt { readTree().size } { panic("Expected tree.")} +} + + +def main() = benchmark(5){run} diff --git a/examples/benchmarks/input_output/number_matrix.check b/examples/benchmarks/input_output/number_matrix.check new file mode 100644 index 000000000..37021f4a2 --- /dev/null +++ b/examples/benchmarks/input_output/number_matrix.check @@ -0,0 +1 @@ +1500 \ No newline at end of file diff --git a/examples/benchmarks/input_output/number_matrix.effekt b/examples/benchmarks/input_output/number_matrix.effekt new file mode 100644 index 000000000..dd3eb3614 --- /dev/null +++ b/examples/benchmarks/input_output/number_matrix.effekt @@ -0,0 +1,90 @@ +import examples/benchmarks/runner + +import io/error +import io/filesystem +import stream +import scanner + + +// number_matrix.txt +// 0 17 34 51 68 +// 13 30 47 64 81 +// 26 43 60 77 94 +// 39 56 73 90 107 +// 52 69 86 103 120 + +def emitDigit(d: Int) = + do emit((d + 48).toByte) + +def emitSpace() = + do emit(32.toByte) + +def emitNewline() = + do emit(10.toByte) + +def emitDigits(n: Int): Unit / emit[Byte] = { + def go(n: Int): Unit = { + if (n < 10) { + emitDigit(n) + } else { + go(n / 10) + emitDigit(n.mod(10)) + } + } + go(n) +} + +def emitNumbers(n: Int) = + for[Int] { range(0, n) } { i => + for[Int] { range(0, n) } { j => + emitDigits((i * 13 + j * 17).mod(256)) + if (j == n - 1) { + emitNewline() + } else { + emitSpace() + } + } + } + +def readDigit(): Int / { Scan[Byte], stop } = { + val byte = do peek[Byte] + if (byte.toInt >= 48 && byte.toInt <= 57) { + do skip[Byte]() + return (byte.toInt - 48) + } else { + do stop() + } +} + +def readDecimal(): Int / Scan[Byte] = { + var result = 0 + for[Int] { many { readDigit() } } { digit => + result = result * 10 + digit + } + result +} + +def readNumbers(): Unit / { Scan[Byte], emit[Int] } = + exhaustively { + do emit(readDecimal()) + do skip[Byte]() + } + + +def run(n: Int): Int = { + with on[IOError].panic; + + val filename = "/tmp/number_matrix.txt" + + val _ = { + with writeFile(filename) + emitNumbers(n) + } + + with readFile(filename) + with returning::scanner[Byte, Int] + sum { readNumbers() } +} + + +def main() = benchmark(5){run} diff --git a/examples/benchmarks/input_output/word_count_ascii.effekt b/examples/benchmarks/input_output/word_count_ascii.effekt index 9f2c0a1df..cfc3df1b9 100644 --- a/examples/benchmarks/input_output/word_count_ascii.effekt +++ b/examples/benchmarks/input_output/word_count_ascii.effekt @@ -4,6 +4,17 @@ import io/error import io/filesystem import stream + +// word_count_ascii.txt +// +// !!!!!!!!!! +// """""""""" +// ########## +// $$$$$$$$$$ +// %%%%%%%%%% +// &&&&&&&&&& +// '''''''''' + record Output(chars: Int, words: Int, lines: Int) def formatWith(output: Output, filename: String): String = diff --git a/examples/benchmarks/input_output/word_count_utf8.effekt b/examples/benchmarks/input_output/word_count_utf8.effekt index 9a15b8297..31786a54f 100644 --- a/examples/benchmarks/input_output/word_count_utf8.effekt +++ b/examples/benchmarks/input_output/word_count_utf8.effekt @@ -5,6 +5,19 @@ import io/filesystem import char import stream + +// word_count_utf8.txt +// πŸ˜€πŸ˜€πŸ˜€πŸ˜€πŸ˜€πŸ˜€πŸ˜€πŸ˜€πŸ˜€πŸ˜€ +// 😁😁😁😁😁😁😁😁😁😁 +// πŸ˜‚πŸ˜‚πŸ˜‚πŸ˜‚πŸ˜‚πŸ˜‚πŸ˜‚πŸ˜‚πŸ˜‚πŸ˜‚ +// πŸ˜ƒπŸ˜ƒπŸ˜ƒπŸ˜ƒπŸ˜ƒπŸ˜ƒπŸ˜ƒπŸ˜ƒπŸ˜ƒπŸ˜ƒ +// πŸ˜„πŸ˜„πŸ˜„πŸ˜„πŸ˜„πŸ˜„πŸ˜„πŸ˜„πŸ˜„πŸ˜„ +// πŸ˜…πŸ˜…πŸ˜…πŸ˜…πŸ˜…πŸ˜…πŸ˜…πŸ˜…πŸ˜…πŸ˜… +// πŸ˜†πŸ˜†πŸ˜†πŸ˜†πŸ˜†πŸ˜†πŸ˜†πŸ˜†πŸ˜†πŸ˜† +// πŸ˜‡πŸ˜‡πŸ˜‡πŸ˜‡πŸ˜‡πŸ˜‡πŸ˜‡πŸ˜‡πŸ˜‡πŸ˜‡ +// 😈😈😈😈😈😈😈😈😈😈 + + record Output(chars: Int, words: Int, lines: Int) def formatWith(output: Output, filename: String): String = diff --git a/examples/stdlib/stream/sum_of_squares.effekt b/examples/stdlib/stream/sum_of_squares.effekt index 7f44ab356..f75e6bd3b 100644 --- a/examples/stdlib/stream/sum_of_squares.effekt +++ b/examples/stdlib/stream/sum_of_squares.effekt @@ -10,6 +10,7 @@ def main() = { val max = 10 println("The sum of squares from 1 to " ++ show(max) ++ " is:") println(sum { + with boundary with limit[Int](max + 1) squares() }) diff --git a/libraries/common/char.effekt b/libraries/common/char.effekt index c005593d6..786c66d7f 100644 --- a/libraries/common/char.effekt +++ b/libraries/common/char.effekt @@ -15,33 +15,57 @@ def isWhitespace(c: Char): Bool = c match { case _ => false } -/// Checks if the given character is an ASCII digit in base 10 -/// Use `digitValue(c: Char)` to get the numeric value out. -def isDigit(c: Char): Bool = { c >= '0' && c <= '9' } - -/// Checks if the given character is an ASCII digit in the given base -/// Use `digitValue(c: Char, base: Int)` to get the numeric value out. -def isDigit(c: Char, base: Int): Bool = { - with on[WrongFormat].default { false } - val _ = digitValue(c, base) - true -} - /// Gets the value of a given ASCII digit in base 10 -def digitValue(c: Char): Int / Exception[WrongFormat] = - digitValue(c, 10) - -/// Gets the value of a given ASCII digit in the given base -def digitValue(c: Char, base: Int): Int / Exception[WrongFormat] = { - val v = c match { - case c and c >= '0' && c <= '9' => (c.toInt - '0'.toInt) - case c and c >= 'a' && c <= 'z' => (c.toInt - 'a'.toInt) + 10 - case c and c >= 'A' && c <= 'Z' => (c.toInt - 'A'.toInt) + 10 - case _ => wrongFormat("'" ++ c.toString ++ "' is not a digit") +def digitValue(char: Char): Option[Int] = + // TODO use fail + if (char >= '0' && char <= '9') { + Some(char.toInt - '0'.toInt) + } else { + None() + } + +/// Gets the value of a given ASCII digit in base 16 +def hexDigitValue(char: Char): Option[Int] = + // TODO use fail + char match { + case char and char >= '0' && char <= '9' => Some(char.toInt - '0'.toInt) + case char and char >= 'A' && char <= 'F' => Some((char.toInt - 'A'.toInt) + 10) + case char and char >= 'a' && char <= 'f' => Some((char.toInt - 'a'.toInt) + 10) + case _ => None() + } + +/// Gets the value of a given ASCII digit in the given base up to 36 +def digitValue(char: Char, base: Int): Option[Int] = { + // TODO use fail + val perhapsDigit = char match { + case char and char >= '0' && char <= '9' => Some(char.toInt - '0'.toInt) + case char and char >= 'A' && char <= 'Z' => Some((char.toInt - 'A'.toInt) + 10) + case char and char >= 'a' && char <= 'z' => Some((char.toInt - 'a'.toInt) + 10) + case _ => None() + } + perhapsDigit match { + case Some(digit) => + if (digit < base) { + Some(digit) + } else { + None() + } + case None() => None() } - if (v >= base) { wrongFormat(c.toString ++ " is not a valid digit in base " ++ base.show) } else { v } } +/// Checks if the given character is an ASCII digit in base 10 +/// Use `digitValue(c: Char)` to get the numeric value out. +def isDigit(char: Char): Bool = digitValue(char).isDefined + +/// Checks if the given character is an ASCII digit in base 16 +/// Use `hexDigitValue(c: Char)` to get the numeric value out. +def isHexDigit(char: Char): Bool = hexDigitValue(char).isDefined + +/// Checks if the given character is an ASCII digit in base 10 +/// Use `digitValue(c: Char)` to get the numeric value out. +def isDigit(char: Char, base: Int): Bool = digitValue(char, base).isDefined + /// Checks if a given character is a 7-bit ASCII character def isASCII(c: Char): Bool = { c.toInt < 128 } diff --git a/libraries/common/effekt.effekt b/libraries/common/effekt.effekt index 65505277f..2ee2de5fa 100644 --- a/libraries/common/effekt.effekt +++ b/libraries/common/effekt.effekt @@ -367,7 +367,11 @@ def pow(base: Double, exponent: Int): Double = { else if (exponent.mod(2) == 0) loop(base * base, exponent / 2, acc) // even exponent else loop(base, exponent - 1, acc * base) // odd exponent } - loop(base, exponent, 1.0) + if (exponent < 0) { + loop(1.0 / base, 0 - exponent, 1.0) + } else { + loop(base, exponent, 1.0) + } } extern pure def pow(base: Double, exponent: Double): Double = diff --git a/libraries/common/json.effekt b/libraries/common/json.effekt new file mode 100644 index 000000000..eeb75ae5d --- /dev/null +++ b/libraries/common/json.effekt @@ -0,0 +1,385 @@ +module json + +import char +import scanner +import stream + +/// for constructing a json value +interface JsonBuilder { + def null(): Unit + def bool(b: Bool): Unit + def number(n: Double): Unit + def string(s: String): Unit + def list[R](){ elements: => R / JsonBuilder }: R + def dict[R](){ elements: => R / JsonObjectBuilder }: R +} +/// for constructing a json object +interface JsonObjectBuilder { + def field[R](k: String){ contents: => R / JsonBuilder }: R +} + +def escape(s: String): String = "\"" ++ s ++ "\"" // TODO handle escapes + +/// Make explicitly bound JsonBuilder instance implicit +def handleJsonBuilder[R]{b: JsonBuilder}{ body: => R / JsonBuilder }: R = try body() with JsonBuilder { + def null() = resume(b.null()) + def bool(v) = resume(b.bool(v)) + def number(n) = resume(b.number(n)) + def string(s) = resume(b.string(s)) + def list() = resume { {c} => b.list{c} } + def dict() = resume { {c} => b.dict{c} } +} +/// Make explicitly bound JsonObjectBuilder instance implicit +def handleJsonObjectBuilder[R]{b: JsonObjectBuilder}{ body: => R / JsonObjectBuilder }: R = try body() with JsonObjectBuilder { + def field(k) = resume { {c} => b.field(k){c} } +} + +// -------------------------------------------------------------------------------- +// Encoding +// -------------------------------------------------------------------------------- +def encodeJsonList[R]{ body: => R / JsonBuilder }: R / emit[String] = { + var first = true + do emit("[") + def c() = { + if (not(first)) { do emit(",") } + first = false + } + val r = encodeJson { + try { body() } with JsonBuilder { + def null() = { c(); resume(do null()) } + def bool(v) = { c(); resume(do bool(v)) } + def number(n) = { c(); resume(do number(n)) } + def string(s) = { c(); resume(do string(s)) } + def list() = resume { { b } => c(); do list{b} } + def dict() = resume { { b } => c(); do dict{b} } + } + } + do emit("]") + r +} +def encodeJsonObject[R]{ body: => R / JsonObjectBuilder }: R / emit[String] = { + var first = true + do emit("{") + def c(k: String) = { + if (not(first)) { do emit(",") } + do emit(escape(k)); do emit(":") + first = false + } + val r = encodeJson { + try body() with JsonObjectBuilder { + def field(k) = resume { {b} => c(k); b() } + } + } + do emit("}") + r +} +/// Main entry point for encoding json. +/// Emits individual tokens of the resulting json. +def encodeJson[R]{ body: => R / JsonBuilder }: R / emit[String] = { + try body() with JsonBuilder { + def null() = { resume(do emit("null")) } + def bool(v) = { resume(do emit( if(v){ "true" } else { "false" } )) } + def number(n) = { resume(do emit(show(n))) } + def string(s) = { resume(do emit(escape(s))) } + def list() = resume { {b} => encodeJsonList { b() } } + def dict() = resume { {b} => encodeJsonObject { b() } } + } +} + +// -------------------------------------------------------------------------------- +// Decoding +// -------------------------------------------------------------------------------- + + +/// Read a double value, in the common format (TODO document) +def readDouble(): Double / Scan[Char] = { + def exponent(pre: Double) = { + try { + if(do peek() == 'e' || do peek() == 'E') { + do skip[Char]() + val e = do peek[Char]() match { + case '+' => do skip[Char](); readDecimal() + case '-' => do skip[Char](); neg(readDecimal()) + case _ => readDecimal() + } + pre * pow(10.0, e) + } else { pre } + } with stop { + pre + } + } + def fraction(pre: Int) = { + if (optionally { readIf('.') }) { + var b = 0.1 + var r = pre.toDouble + while (optionally[Int] { readDigit() } is Some(d)) { + r = r + b * d.toDouble + b = b * 0.1 + } + exponent(r) + } else { + pre.toDouble + } + } + fraction(readInteger()) +} + +def expectString(string: String): Unit / { Scan[Char], Exception[WrongFormat] } = + for[Char] { string.each } { char => + expect[Unit]("Expected " ++ string) { readIf(char) } + } + +/// Read and unescape a string in "" +def readQuotedString(): Unit / { Scan[Char], emit[Char], Exception[WrongFormat] } = { + try { + skipWhitespace() + expect[Unit]("Expected \"") { readIf('"') } + while(read[Char]() is c and c != '"') { + c match { + case '\\' => read[Char]() match { + case '"' => do emit('\"') + case '\\' => do emit('\\') + case '/' => do emit('/') + case 'b' => <> + case 'f' => <> + case 'n' => do emit('\n') + case 'r' => do emit('\r') + case 't' => do emit('\t') + case c => wrongFormat("Invalid escape sequence '\\" ++ c.toString ++ "'") + } + case o => do emit(o) + } + } + } with stop { + wrongFormat("Unexpected end of input while reading a string") + } +} + +/// Decode a json string and do the appropriate calls to an implicitly bound JsonBuilder +def decodeJson(): Unit / {Scan[Char], JsonBuilder, Exception[WrongFormat]} = { + with boundary + skipWhitespace() + if(do peek[Char]() is c) { + c match { + case 'n' => expectString("null"); do null() + case 't' => expectString("true"); do bool(true) + case 'f' => expectString("false"); do bool(false) + case '-' => do number(readDouble()) + case d and d.isDigit => do number(readDouble()) + case '"' => do string(collectString { readQuotedString() }) + case '{' => do dict{ decodeJsonObject() } + case '[' => do list{ decodeJsonList() } + case _ => println("Unexpected " ++ c.toString); <> + } + } +} +def decodeJsonObject(): Unit / {Scan[Char], JsonObjectBuilder, Exception[WrongFormat]} = { + var first = true + expectString("{") + with boundary + skipWhitespace() + while(do peek[Char]() is c and (c != '}')) { + if (not(first)) { expectString(",") } + val k: String = collectString { readQuotedString() } + skipWhitespace() + expectString(":") + do field(k){ + decodeJson() + } + skipWhitespace() + first = false + } + do skip[Char]() +} +def decodeJsonList(): Unit / {Scan[Char], JsonBuilder, Exception[WrongFormat]} = { + var first = true + expectString("[") + with boundary + skipWhitespace() + while(do peek[Char]() is c and (c != ']')) { + if (not(first)) { expectString(",") } + decodeJson() + skipWhitespace() + first = false + } + do skip[Char]() +} + +// -------------------------------------------------------------------------------- +// Ignoring +// -------------------------------------------------------------------------------- +/// Ignore the generated calls to JsonObjectBuilder +def ignoreDict[R](){ body: => R / JsonObjectBuilder }: R = try body() with JsonObjectBuilder { + def field(k) = resume { {v} => ignore{v} } +} + +/// Ignore the generated calls to JsonBuilder +def ignore[R](){ body: => R / JsonBuilder }: R = try body() with JsonBuilder { + def number(n) = resume(()) + def bool(b) = resume(()) + def null() = resume(()) + def string(s) = resume(()) + def list() = resume { {e} => ignore{e} } + def dict() = resume { {e} => ignoreDict{e} } +} + +// -------------------------------------------------------------------------------- +// Residualized +// -------------------------------------------------------------------------------- +/// Residualized Json value +type JsonValue { + Number(n: Double); + Bool(b: Bool); + Null(); + String(s: String); + List(l: List[JsonValue]); + Dict(els: List[(String, JsonValue)]) +} + +// -------------------------------------------------------------------------------- +// Residualized --> Effect +// -------------------------------------------------------------------------------- +/// Emit the appropriate calls to JsonBuilder to recreate the given JsonValue +def unbuild(v: JsonValue): Unit / JsonBuilder = v match { + case Number(n) => do number(n) + case Bool(b) => do bool(b) + case Null() => do null() + case String(s) => do string(s) + case List(l) => do list { + l.foreach { e => unbuild(e) } + } + case Dict(els) => do dict { + els.foreach { + case (k, v) => do field(k){ unbuild(v) } + } + } +} + +// -------------------------------------------------------------------------------- +// Effect --> Residualized +// -------------------------------------------------------------------------------- +def build[R](){ body: => R / JsonBuilder }: (R, JsonValue) = { + var r = Null() + val x = try body() with JsonBuilder { + def number(n) = { r = Number(n); resume(()) } + def bool(b) = { r = Bool(b); resume(()) } + def null() = { r = Null(); resume(()) } + def string(s) = { r = String(s); resume(()) } + def list() = resume { {els} => + val x = buildList {els} + r = List(x.second) + x.first + } + def dict() = resume { {fs} => + val x = buildDict {fs} + r = Dict(x.second) + x.first + } + } + (x, r) +} +def buildList[R](){ body: => R / JsonBuilder }: (R, List[JsonValue]) = collectList[JsonValue, R] { + try body() with JsonBuilder { + def number(n) = { do emit(Number(n)); resume(()) } + def bool(b) = { do emit(Bool(b)); resume(()) } + def null() = { do emit(Null()); resume(()) } + def string(s) = { do emit(String(s)); resume(()) } + def list() = resume { {els} => + val x = buildList {els} + do emit(List(x.second)) + x.first + } + def dict() = resume { {fs} => + val x = buildDict {fs} + do emit(Dict(x.second)) + x.first + } + } +} +def buildDict[R](){ body: => R / JsonObjectBuilder }: (R, List[(String, JsonValue)]) = collectList[(String, JsonValue), R] { + try body() with JsonObjectBuilder { + def field(k) = resume { {v} => + val x = build{v} + do emit((k, x.second)) + x.first + } + } +} + +// -------------------------------------------------------------------------------- +// Examples +// -------------------------------------------------------------------------------- +namespace test { + def main() = { + with on[WrongFormat].panic + + // Read quoted string + feed("\"\ta\n\ra\"") { + with scanner[Char, Unit] + println(collectString { readQuotedString() }) + } + + // Parse example + feed("""{ "a": null, "b": [true,false,false,true], "f": 12.532 }"""){ + with scanner[Char] + + def k = new JsonObjectBuilder { + def field(k){v} = { println(k); ignore{v} } + } + def d = new JsonBuilder { + def number(n) = println(n) + def bool(b) = () + def null() = println("NULL") + def string(s) = () + def list(){e} = handleJsonBuilder{d}{e} + def dict(){e} = handleJsonObjectBuilder{k}{e} + } + handleJsonBuilder{d}{ decodeJson() } + } + try{ + + // Encode example + encodeJson { + do dict{ + do field("x"){ + do string("Hallo") + } + do field("y"){ + do null() + } + do field("z"){ + do number(12.3783) + } + } + } + + // format with intermediate value + val j = feed("""{ + "a": null, + "b": [true, false, false, true], "f": 12.532 + } """){ + with scanner[Char, JsonValue] + build { + decodeJson() + }.second + } + encodeJson{ + unbuild(j) + } + + // format (minify) example + encodeJson{ + feed("""{ + "a": null, + "b": [true, false, false, true], "f": 12.532 + } """){ + with scanner[Char, Unit] + decodeJson() + } + } + + } with emit[String] { e => + resume(println(e)) + } + } +} \ No newline at end of file diff --git a/libraries/common/scanner.effekt b/libraries/common/scanner.effekt new file mode 100644 index 000000000..97e0da243 --- /dev/null +++ b/libraries/common/scanner.effekt @@ -0,0 +1,171 @@ +module scanner + +import char +import stream + +interface Scan[A] { + /// Return the next character, not advancing. + /// That is, this does not change what any future calls on the Scanner return. + def peek(): A / stop + /// Advance the Scanner to the next character. + def skip(): Unit / stop +} + +/// Advance the Scanner to the next character, returning it. +def read[A](): A / { Scan[A], stop } = { + val t = do peek[A]() + do skip[A]() + return t +} + +/// Run a scanner by reading from an input stream, discarding its output. +def scanner[A] { scanner: () => Unit / Scan[A] }: Unit / read[A] = + returning::scanner[A, Unit]{scanner} + +def expect(message: String) { scanner: () => Unit / stop }: Unit / Exception[WrongFormat] = + returning::expect[Unit](message){scanner} + +/// Check that the next token satisfies the predicate and skip and return it when it does. +def readIf[A] { predicate: A => Bool }: A / { Scan[A], stop } = { + val t = do peek[A]() + if (predicate(t)) { + do skip[A]() + return t + } else { + do stop() + } +} + +/// Check that the next token satisfies the predicate and skip it when it does. +def skipIf[A] { predicate: A => Bool }: Unit / { Scan[A], stop } = { + readIf{predicate}; () +} + +def readSome[A, B] { convert: A => Option[B] }: B / { Scan[A], stop } = + if (convert(do peek()) is Some(t)) { + do skip[A]() + return t + } else { + do stop() + } + +/// Reads until the predicate does not hold for the next token. +/// Emits the tokens read. +def readWhile[A] { predicate: A => Bool }: Unit / { Scan[A], emit[A] } = + many { readIf{predicate} } + +/// Skips until the predicate does not hold for the next token. +def skipWhile[A] { predicate: A => Bool }: Unit / { Scan[A] } = + for[A] { readWhile{predicate} } { _ => () } + + +/// Read the next character if it is the given one and return true in this case. +def readIf(e: Char): Unit / { Scan[Char], stop } = { + readIf[Char] { c => c == e }; () +} + +/// Check that the next token satisfies the predicate and skip it when it does. +def skipIf(e: Char): Unit / { Scan[Char], stop } = + skipIf { c => c == e } + +/// Skip until the next character is not a whitespace character +def skipWhitespace(): Unit / Scan[Char] = + skipWhile { c => c.isWhitespace } + +/// Read as many characters corresponding to the given string as possible. +def readString(string: String): Unit / { Scan[Char], stop } = + for[Char] { string.each } { char => readIf(char) } + +/// Check that the next character is a digit in base 10, and if so read and return it. +def readDigit(): Int / { Scan[Char], stop } = + readSome[Char, Int]{ char => digitValue(char) } + +/// Read a positive decimal number. +def readDecimal(): Int / Scan[Char] = { + var result = 0 + for[Int] { many { readDigit() } } { digit => + result = result * 10 + digit + } + result +} + +/// Check that the next character is a digit in base 16, and if so read and return it. +def readHexDigit(): Int / { Scan[Char], stop } = + readSome[Char, Int]{ char => hexDigitValue(char) } + +/// Read a hexadecimal number. +def readHexadecimal(): Int / Scan[Char] = { + var result = 0 + for[Int] { many { readHexDigit() } } { digit => + result = result * 16 + digit + } + result +} + +/// Read a decimal integer. +def readInteger(): Int / Scan[Char] = + if (optionally { readIf('-') }) { + neg(readDecimal()) + } else { + readDecimal() + } + + +namespace returning { + +/// Run a scanner by reading from an input stream. +def scanner[A, R] { scanner: () => R / Scan[A] }: R / read[A] = { + var last = None() + try { + scanner() + } with Scan[A] { + def peek() = resume { + last match { + case None() => + val t = do read[A]() + last = Some(t) + return t + case Some(t) => + return t + } + } + def skip() = resume { + last match { + case None() => + val _ = do read[A]() + return () + case Some(t) => + last = None() + return () + } + } + } +} + +def expect[R](message: String) { scanner: () => R / stop }: R / Exception[WrongFormat] = + attempt { scanner() } { wrongFormat(message) } + +} + + +namespace test { + def main(): Unit = { + + val input = + """ + 12 + -13 + A1B2 + """ + + with feed(input) + with scanner[Char] + + skipWhitespace() + println(readInteger().show) + skipWhitespace() + println(readInteger().show) + skipWhitespace() + println(readHexadecimal().show) + } +} diff --git a/libraries/common/stream.effekt b/libraries/common/stream.effekt index 6337c1f1c..0c16cdb1c 100644 --- a/libraries/common/stream.effekt +++ b/libraries/common/stream.effekt @@ -19,24 +19,13 @@ effect read[A](): A / stop /// Signal from a producer to a consumer that there are no further values. effect stop(): Nothing -/// Canonical handler of push streams that performs `action` for every -/// value emitted by `stream`. -def for[A, R] { stream: () => R / emit[A] } { action: A => Unit }: R = - try { - stream() - } with emit[A] { value => - resume(action(value)) - } - /// Like `for[A, R]`, but ignores the result of the stream, and consequently /// works for any type. Use this to annotate the type of stream elements /// `A` without having to also annotate `R`. /// /// e.g. for[Int] { prog() } { el => println(el) } -def for[A] { stream: () => Any / emit[A] } { action: A => Unit }: Unit = { - for[A, Any]{stream}{action} - () -} +def for[A] { stream: () => Unit / emit[A] } { action: A => Unit }: Unit = + returning::for[A, Unit]{stream}{action} /// Turns a `list` into a producer of a push stream /// by emitting each contained value left-to-right. @@ -81,20 +70,13 @@ def each(bytes: ByteArray): Unit / emit[Byte] = { // not seq -def boundary[R] { program: () => R / stop }: Option[R] = - try { - Some(program()) - } with stop { - None() - } - -def boundary { program: () => Any / stop }: Unit = { - boundary[Any]{program} +def boundary { program: () => Unit / stop }: Unit = { + returning::boundary[Unit]{program} () } /// Run `program` forever until `stop` is thrown. -def exhaustively { program: () => Any / stop }: Unit = +def exhaustively { program: () => Unit / stop }: Unit = try { def go(): Unit = { program() @@ -105,6 +87,24 @@ def exhaustively { program: () => Any / stop }: Unit = () } +def many[A] { action: () => A / stop }: Unit / emit[A] = + exhaustively { do emit(action()) } + +def some[A] { action: () => A / stop }: Unit / { emit[A], stop } = { + do emit(action()) + many { action() } +} + +def optionally { program: () => Unit / stop }: Bool = + returning::optionally[Unit]{program}.isDefined + +def attempt[R] { program: () => R / stop } { fallback: () => R }: R = + try { + program() + } with stop { + fallback() + } + /// In Effekt lower bounds are inclusive and upper bounds are exclusive record Indexed[A](index: Int, value: A) @@ -123,39 +123,13 @@ def rangeFrom(lower: Int): Unit / emit[Int] = { rangeFrom(lower + 1) } +def index[A] { stream: () => Unit / emit[A] }: Unit / emit[Indexed[A]] = + returning::index[A, Unit]{stream} -def index[A, R] { stream: () => R / emit[A] }: R / emit[Indexed[A]] = { - var i = 0; - try { - stream() - } with emit[A] { v => - val c = i; - i = c + 1; - resume(do emit(Indexed(c, v))) - } -} /// If `number` is zero or negative it does nothing -def limit[A, R](number: Int) { stream: () => R / emit[A] }: R / { emit[A], stop } = { - if (number <= 0) do stop(); - var i = 1; - try { - stream() - } with emit[A] { v => - if (i < number) { - i = i + 1; - resume(do emit(v)) - } else { - do stop() - } - } -} - -/// If `number` is zero or negative it does nothing -def limit[A](number: Int) { stream: () => Any / emit[A] }: Unit / emit[A] = - boundary { - limit[A, Any](number){stream} - } +def limit[A](number: Int) { stream: () => Unit / emit[A] }: Unit / {emit[A], stop} = + returning::limit[A, Unit](number){stream} /// If `number` is zero or negative it does nothing @@ -166,61 +140,17 @@ def replicate[A](number: Int) { action: () => A }: Unit / emit[A] = } -def sum[R] { stream : () => R / emit[Int] }: (R, Int) = { - var s = 0; - try { - (stream(), s) - } with emit[Int] { v => - s = s + v; - resume(()) - } -} +def sum { stream : () => Unit / emit[Int] }: Int = + returning::sum[Unit]{stream}.second -def sum { stream : () => Any / emit[Int] }: Int = - sum[Any]{stream}.second +def collectList[A] { stream: () => Unit / emit[A] }: List[A] = + returning::collectList[A, Unit]{stream}.second -def collectList[A, R] { stream: () => R / emit[A] }: (R, List[A]) = - try { - (stream(), Nil()) - } with emit[A] { (v) => - val (r, vs) = resume(()); - (r, Cons(v, vs)) - } - -def collectList[A] { stream: () => Any / emit[A] }: List[A] = - collectList[A, Any]{stream}.second - -def collectArray[A, R] { stream: () => R / emit[A] }: (R, Array[A]) = { - var i = 0 - var a = array::allocate(1) - try { - (stream(), a.resize(i)) - } with emit[A] { (v) => - if (i >= a.size) { a = a.resize(2 * a.size) } - a.unsafeSet(i, v) - i = i + 1 - resume(()) - } -} - -def collectArray[A] { stream: () => Any / emit[A] }: Array[A] = - collectArray[A, Any]{stream}.second - -def collectBytes[R] { stream: () => R / emit[Byte] }: (R, ByteArray) = { - var i = 0 - var a = bytearray::allocate(1) - try { - (stream(), a.resize(i)) - } with emit[Byte] { (v) => - if (i >= a.size) { a = a.resize(2 * a.size) } - a.unsafeSet(i, v) - i = i + 1 - resume(()) - } -} +def collectArray[A] { stream: () => Unit / emit[A] }: Array[A] = + returning::collectArray[A, Unit]{stream}.second -def collectBytes { stream: () => Any / emit[Byte] }: ByteArray = - collectBytes[Any]{stream}.second +def collectBytes { stream: () => Unit / emit[Byte] }: ByteArray = + returning::collectBytes[Unit]{stream}.second def feed[T, R](list: List[T]) { reader: () => R / read[T] }: R = { var l = list @@ -272,34 +202,12 @@ def feed[R](bytes: ByteArray) { reader: () => R / read[Byte] }: R = { } } -def source[A, R] { stream: () => Any / emit[A] } { reader: () => R / read[A] }: R = { - var next = box { None() } - next = box { - try { - stream() - None() - } with emit[A] { (v) => - next = box { resume(()) } - Some(v) - } - } - try { - reader() - } with read[A] { - resume { - next().getOrElse { do stop() } - } - } -} - -def source[A] { stream: () => Any / emit[A] } { reader: () => Any / read[A] }: Unit = { - source[A, Any]{stream}{reader} - () -} +def source[A] { stream: () => Unit / emit[A] } { reader: () => Unit / read[A] }: Unit = + returning::source[A, Unit]{stream}{reader} /// Combines two streams together producing a stream of pairs in lockstep. -/// Given two streams of length `n` and `m`, it produces a stream of length `min(n, m)`. -def zip[A, B] { stream1: () => Any / emit[A] } { stream2: () => Any / emit[B] } { action: (A, B) => Any }: Unit = { +/// Terminates when either of them terminates. +def zip[A, B] { stream1: () => Unit / emit[A] } { stream2: () => Unit / emit[B] } { action: (A, B) => Unit }: Unit = { with source[A] { stream1 } with source[B] { stream2 } @@ -423,7 +331,7 @@ def encodeUTF8[R] { stream: () => R / emit[Char] }: R / emit[Byte] = resume(encodeChar(char)) } -def feed(string: String) { reader: () => Unit / read[Char] } = +def feed[R](string: String) { reader: () => R / read[Char] } = feed(string.fromString) { decodeUTF8 { reader() @@ -437,10 +345,129 @@ def each(string: String): Unit / emit[Char] = } } +def collectString { stream: () => Unit / emit[Char] }: String = + returning::collectString[Unit]{stream}.second + +namespace returning { + +/// Canonical handler of push streams that performs `action` for every +/// value emitted by `stream`. +def for[A, R] { stream: () => R / emit[A] } { action: A => Unit }: R = + try { + stream() + } with emit[A] { value => + resume(action(value)) + } + +def boundary[R] { program: () => R / stop }: Option[R] = + try { + Some(program()) + } with stop { + None() + } + +def optionally[R] { program: () => R / stop }: Option[R] = + try { + Some(program()) + } with stop { + None() + } + +def index[A, R] { stream: () => R / emit[A] }: R / emit[Indexed[A]] = { + var i = 0; + try { + stream() + } with emit[A] { v => + val c = i; + i = c + 1; + resume(do emit(Indexed(c, v))) + } +} + +/// If `number` is zero or negative it does nothing +def limit[A, R](number: Int) { stream: () => R / emit[A] }: R / { emit[A], stop } = { + if (number <= 0) do stop(); + var i = 1; + try { + stream() + } with emit[A] { v => + if (i < number) { + i = i + 1; + resume(do emit(v)) + } else { + do stop() + } + } +} + +def sum[R] { stream : () => R / emit[Int] }: (R, Int) = { + var s = 0; + try { + (stream(), s) + } with emit[Int] { v => + s = s + v; + resume(()) + } +} + +def collectList[A, R] { stream: () => R / emit[A] }: (R, List[A]) = + try { + (stream(), Nil()) + } with emit[A] { (v) => + val (r, vs) = resume(()); + (r, Cons(v, vs)) + } + +def collectArray[A, R] { stream: () => R / emit[A] }: (R, Array[A]) = { + var i = 0 + var a = array::allocate(1) + try { + (stream(), a.resize(i)) + } with emit[A] { (v) => + if (i >= a.size) { a = a.resize(2 * a.size) } + a.unsafeSet(i, v) + i = i + 1 + resume(()) + } +} + +def collectBytes[R] { stream: () => R / emit[Byte] }: (R, ByteArray) = { + var i = 0 + var a = bytearray::allocate(1) + try { + (stream(), a.resize(i)) + } with emit[Byte] { (v) => + if (i >= a.size) { a = a.resize(2 * a.size) } + a.unsafeSet(i, v) + i = i + 1 + resume(()) + } +} + def collectString[R] { stream: () => R / emit[Char] }: (R, String) = { val (result, bytes) = collectBytes[R] { encodeUTF8 { stream } } (result, bytes.toString) } -def collectString { stream: () => Any / emit[Char] }: String = - collectString[Any]{stream}.second +def source[A, R] { stream: () => Unit / emit[A] } { reader: () => R / read[A] }: R = { + var next = box { None() } + next = box { + try { + stream() + next = box { None() } + None() + } with emit[A] { (v) => + next = box { resume(()) } + Some(v) + } + } + try { + reader() + } with read[A] { + resume { + next().getOrElse { do stop() } + } + } +} + +} From d70540488440a9dc99de45aca869530026abf2c1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jonathan=20Immanuel=20Brachth=C3=A4user?= Date: Sat, 21 Dec 2024 11:47:34 +0100 Subject: [PATCH 07/31] Update/kiama lsp (#753) Update kiama to support LSP notebooks --- build.sbt | 15 ++++++++-- effekt/jvm/src/main/scala/effekt/Server.scala | 28 ++----------------- kiama | 2 +- 3 files changed, 15 insertions(+), 30 deletions(-) diff --git a/build.sbt b/build.sbt index a4dd03c09..08cf17b7b 100644 --- a/build.sbt +++ b/build.sbt @@ -47,8 +47,7 @@ lazy val replDependencies = Seq( ) lazy val lspDependencies = Seq( - "org.eclipse.lsp4j" % "org.eclipse.lsp4j" % "0.12.0", - "com.google.code.gson" % "gson" % "2.8.9" + "org.eclipse.lsp4j" % "org.eclipse.lsp4j" % "0.23.1" ) lazy val testingDependencies = Seq( @@ -63,7 +62,8 @@ lazy val kiama: CrossProject = crossProject(JSPlatform, JVMPlatform).in(file("ki name := "kiama" ) .jvmSettings( - libraryDependencies ++= (replDependencies ++ lspDependencies) + libraryDependencies ++= (replDependencies ++ lspDependencies ++ testingDependencies), + testFrameworks += new TestFramework("utest.runner.Framework") ) lazy val root = project.in(file("effekt")) @@ -116,6 +116,15 @@ lazy val effekt: CrossProject = crossProject(JSPlatform, JVMPlatform).in(file("e assembly / assemblyJarName := "effekt.jar", + // there is a conflict between the two transitive dependencies "gson:2.11.0" + // and "error_prone_annotations:2.27.0", so we need the merge strategy here for `sbt install` + assembly / assemblyMergeStrategy := { + case PathList("META-INF", "versions", "9", "module-info.class") => MergeStrategy.first + case x => + val oldStrategy = (assembly / assemblyMergeStrategy).value + oldStrategy(x) + }, + // we use the lib folder as resource directory to include it in the JAR Compile / unmanagedResourceDirectories += (ThisBuild / baseDirectory).value / "libraries", diff --git a/effekt/jvm/src/main/scala/effekt/Server.scala b/effekt/jvm/src/main/scala/effekt/Server.scala index ac875da49..12d89b88f 100644 --- a/effekt/jvm/src/main/scala/effekt/Server.scala +++ b/effekt/jvm/src/main/scala/effekt/Server.scala @@ -5,11 +5,9 @@ import effekt.core.PrettyPrinter import effekt.source.{ FunDef, Hole, ModuleDecl, Tree } import effekt.util.{ PlainMessaging, getOrElseAborting } import effekt.util.messages.EffektError - -import kiama.util.{ Filenames, Position, Services, Source } +import kiama.util.{ Filenames, Notebook, NotebookCell, Position, Services, Source } import kiama.output.PrettyPrinterTypes.Document - -import org.eclipse.lsp4j.{ Diagnostic, DocumentSymbol, SymbolKind, ExecuteCommandParams } +import org.eclipse.lsp4j.{ Diagnostic, DocumentSymbol, ExecuteCommandParams, SymbolKind } /** * effekt.Intelligence <--- gathers information -- LSPServer --- provides LSP interface ---> kiama.Server @@ -95,28 +93,6 @@ trait LSPServer extends kiama.util.Server[Tree, EffektConfig, EffektError] with info <- getHoleInfo(tree)(context) } yield info - // The implementation in kiama.Server does not support file sources - def toURI(filename: String): String = { - if (filename startsWith "file://") - filename - else - if (filename startsWith "./") - "file://" ++ Filenames.cwd() ++ "/" ++ Filenames.dropPrefix(filename, ".") - else - s"file://$filename" - } - - override def locationOfNode(node: Tree): Location = { - (positions.getStart(node), positions.getFinish(node)) match { - case (start @ Some(st), finish @ Some(_)) => - val s = convertPosition(start) - val f = convertPosition(finish) - new Location(toURI(st.source.name), new LSPRange(s, f)) - case _ => - null - } - } - def positionToLocation(p: Position): Location = { val s = convertPosition(Some(p)) new Location(p.source.name, new LSPRange(s, s)) diff --git a/kiama b/kiama index ab6aef78f..5f5b19950 160000 --- a/kiama +++ b/kiama @@ -1 +1 @@ -Subproject commit ab6aef78fc4ce99a9ad2db8555718d00b03f7b8a +Subproject commit 5f5b19950563dc1ddb1fa4c5343c00870dd1f359 From 9007b059291153c7d9279ec729d62803a7b47275 Mon Sep 17 00:00:00 2001 From: Can <32599057+CanCodes@users.noreply.github.com> Date: Sat, 21 Dec 2024 12:54:01 +0100 Subject: [PATCH 08/31] Add 'show' for 'Byte' on JavaScript (#752) Fixes #749 --- libraries/common/effekt.effekt | 1 + 1 file changed, 1 insertion(+) diff --git a/libraries/common/effekt.effekt b/libraries/common/effekt.effekt index 2ee2de5fa..5fb10bea9 100644 --- a/libraries/common/effekt.effekt +++ b/libraries/common/effekt.effekt @@ -85,6 +85,7 @@ extern pure def show(value: Char): String = """ extern pure def show(value: Byte): String = + js "'' + ${value}" llvm """ %z = call %Pos @c_bytearray_show_Byte(i8 ${value}) ret %Pos %z From 69471e536b926d8c9c96bd27c895b7a52e06a9fa Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jonathan=20Immanuel=20Brachth=C3=A4user?= Date: Sat, 21 Dec 2024 21:10:29 +0100 Subject: [PATCH 09/31] Fixes #733 by bringing the rewritten definitions into scope, instead of old ones (#754) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit See discussion in #733 for how I traced down the bug. Summary: At the point where we dealias the binding `l = m`, the core tree looks like: ``` let l = m; def b_k_1(r, xs) = // note the l here: let v_3 = run {link(k, v, l, r)} go(bitwiseShl225(level, 1), v_3, xs); ... ``` As it can be clearly seen, `l` is used in the join points `b_k_...`. The bindings, after rewriting look like this. ``` let l = m; def b_k_1(r, xs) = // note the m here: let v_3 = run {link(k, v, m, r)} go(bitwiseShl225(level, 1), v_3, xs); ... ``` It appears the renaming was successful. However, the rewritten defs are updated in the `InlineContext`, which can be seen by inspecting `rewrite(List[Definition])` https://github.com/effekt-lang/effekt/blob/9007b059291153c7d9279ec729d62803a7b47275/effekt/shared/src/main/scala/effekt/core/Inline.scala#L88C7-L97 and its callsite: https://github.com/effekt-lang/effekt/blob/9007b059291153c7d9279ec729d62803a7b47275/effekt/shared/src/main/scala/effekt/core/Inline.scala#L124-L126 So, the next time a `def` is inlined, the old `def` is inlined. It still refers to the old, aliased but now removed, binding... Hence causing the error. In this PR, I update the defs in the context to now contain the rewritten (dealiased) right hand sides. --------- Co-authored-by: JiΕ™Γ­ BeneΕ‘ --- .../test/scala/effekt/ChezSchemeTests.scala | 5 +- .../jvm/src/test/scala/effekt/LLVMTests.scala | 3 + .../src/main/scala/effekt/core/Inline.scala | 16 +- examples/pos/issue733.check | 0 examples/pos/issue733.effekt | 259 ++++++++++++++++++ 5 files changed, 278 insertions(+), 5 deletions(-) create mode 100644 examples/pos/issue733.check create mode 100644 examples/pos/issue733.effekt diff --git a/effekt/jvm/src/test/scala/effekt/ChezSchemeTests.scala b/effekt/jvm/src/test/scala/effekt/ChezSchemeTests.scala index 89a0b0eee..bf89f694c 100644 --- a/effekt/jvm/src/test/scala/effekt/ChezSchemeTests.scala +++ b/effekt/jvm/src/test/scala/effekt/ChezSchemeTests.scala @@ -56,9 +56,12 @@ abstract class ChezSchemeTests extends EffektTests { examplesDir / "pos" / "io", // async io is only implemented for monadic JS - examplesDir / "pos" / "genericcompare.effekt", // genericCompare is only implemented for JS examplesDir / "pos" / "issue429.effekt", + + // Generic comparison + examplesDir / "pos" / "genericcompare.effekt", + examplesDir / "pos" / "issue733.effekt", ) } diff --git a/effekt/jvm/src/test/scala/effekt/LLVMTests.scala b/effekt/jvm/src/test/scala/effekt/LLVMTests.scala index 6359f5f30..a65046cd0 100644 --- a/effekt/jvm/src/test/scala/effekt/LLVMTests.scala +++ b/effekt/jvm/src/test/scala/effekt/LLVMTests.scala @@ -96,6 +96,9 @@ class LLVMTests extends EffektTests { // Generic equality examplesDir / "pos" / "issue429.effekt", + + // Generic comparison + examplesDir / "pos" / "issue733.effekt", ) override lazy val ignored: List[File] = bugs ++ missingFeatures diff --git a/effekt/shared/src/main/scala/effekt/core/Inline.scala b/effekt/shared/src/main/scala/effekt/core/Inline.scala index 18f91c607..8fc01284e 100644 --- a/effekt/shared/src/main/scala/effekt/core/Inline.scala +++ b/effekt/shared/src/main/scala/effekt/core/Inline.scala @@ -31,6 +31,8 @@ object Inline { ) { def ++(other: Map[Id, Definition]): InlineContext = InlineContext(usage, defs ++ other, maxInlineSize, inlineCount) + def ++(other: List[Definition]): InlineContext = ++(other.map(d => d.id -> d).toMap) + def ++=(fresh: Map[Id, Usage]): Unit = { usage ++= fresh } } @@ -85,8 +87,15 @@ object Inline { def used(id: Id)(using ctx: InlineContext): Boolean = ctx.usage.isDefinedAt(id) + /** + * Rewrites the list of definition and returns: + * 1. the updated list + * 2. definitions + * a. original defnitions: in case we need to dealias elsewhere + * b. the updated definitions, where the rhs might have been dealiased already (see #733) + */ def rewrite(definitions: List[Definition])(using ctx: InlineContext): (List[Definition], InlineContext) = - given allDefs: InlineContext = ctx ++ definitions.map(d => d.id -> d).toMap + given allDefs: InlineContext = ctx ++ definitions val filtered = definitions.collect { case Definition.Def(id, block) => Definition.Def(id, rewrite(block)) @@ -94,12 +103,11 @@ object Inline { case Definition.Let(id, tpe, binding) if !binding.isInstanceOf[ValueVar] => Definition.Let(id, tpe, rewrite(binding)) } - (filtered, allDefs) + (filtered, allDefs ++ filtered) def blockDefFor(id: Id)(using ctx: InlineContext): Option[Block] = ctx.defs.get(id) map { - // TODO rewriting here leads to a stack overflow in one test, why? - case Definition.Def(id, block) => block //rewrite(block) + case Definition.Def(id, block) => block case Definition.Let(id, _, binding) => INTERNAL_ERROR("Should not happen") } diff --git a/examples/pos/issue733.check b/examples/pos/issue733.check new file mode 100644 index 000000000..e69de29bb diff --git a/examples/pos/issue733.effekt b/examples/pos/issue733.effekt new file mode 100644 index 000000000..7219f3b07 --- /dev/null +++ b/examples/pos/issue733.effekt @@ -0,0 +1,259 @@ +effect emit[A](value: A): Unit + +namespace newset { + record Set[A](internal: newmap::Map[A, Unit]) + + def empty[A](): Set[A] = { + Set(newmap::empty()) + } + + def fromList[A](list: List[A]): Set[A] = { + val m: newmap::Map[A, Unit] = list.map { k => (k, ()) }.newmap::fromList + Set(m) + } +} + +namespace newmap { + type Map[K, V] { + Bin(size: Int, k: K, v: V, left: Map[K, V], right: Map[K, V]); + Tip() + } + + def empty[K, V](): Map[K, V] = { + Tip() + } + + def singleton[K, V](k: K, v: V): Map[K, V] = { + Bin(1, k, v, Tip(), Tip()) + } + + def size[K, V](m: Map[K, V]): Int = { + m match { + case Tip() => 0 + case Bin(size, _, _, _, _) => size + } + } + + val ratio = 2 + val delta = 3 + + def bin[K, V](k: K, v: V, l: Map[K, V], r: Map[K, V]): Map[K, V] = { + Bin(l.size() + r.size() + 1, k, v, l, r) + } + + def balance[K, V](k: K, v: V, l: Map[K, V], r: Map[K, V]): Map[K, V] = { + def singleL[A, B](k1: A, v1: B, t1: Map[A, B], m: Map[A, B]): Map[A, B] = { + m match { + case Bin(_, k2, v2, t2, t3) => bin(k2, v2, bin(k1, v1, t1, t2), t3) + case _ => <{ "impossible: singleL: Tip" }> + } + } + + def singleR[A, B](k1: A, v1: B, m: Map[A, B], t3: Map[A, B]): Map[A, B] = { + m match { + case Bin(_, k2, v2, t1, t2) => bin(k2, v2, t1, bin(k1, v1, t2, t3)) + case _ => <{ "impossible: singleR: Tip" }> + } + } + + def doubleL[A, B](k1: A, v1: B, t1: Map[A, B], m: Map[A, B]): Map[A, B] = { + m match { + case Bin(_, k2, v2, Bin(_, k3, v3, t2, t3), t4) => + bin(k3, v3, bin(k1, v1, t1, t2), bin(k2, v2, t3, t4)) + case _ => <{ "impossible: doubleL: Tip" }> + } + } + + def doubleR[A, B](k1: A, v1: B, m: Map[A, B], t4: Map[A, B]): Map[A, B] = { + m match { + case Bin(_, k2, v2, t1, Bin(_, k3, v3, t2, t3)) => + bin(k3, v3, bin(k2, v2, t1, t2), bin(k1, v1, t3, t4)) + case _ => + <{ "impossible: doubleR: Tip" }> + } + } + + def rotateL[A, B](k: A, v: B, l: Map[A, B], r: Map[A, B]): Map[A, B] = { + r match { + case Bin(_, _, _, rl, rr) and (rl.size() < ratio * rr.size()) => singleL(k, v, l, r) + case _ => doubleL(k, v, l, r) + } + } + def rotateR[A, B](k: A, v: B, l: Map[A, B], r: Map[A, B]): Map[A, B] = { + l match { + case Bin(_, _, _, ll, lr) and (lr.size() < ratio * ll.size()) => singleR(k, v, l, r) + case _ => doubleR(k, v, l, r) + } + } + + val sizeL = l.size() + val sizeR = r.size() + val sizeCombined = sizeL + sizeR + 1 + + if ((sizeL + sizeR) <= 1) { Bin(sizeCombined, k, v, l, r) } + else if (sizeR > (delta * sizeL)) { rotateL(k, v, l, r) } + else if (sizeL > (delta * sizeR)) { rotateR(k, v, l, r) } + else { Bin(sizeCombined, k, v, l, r)} + } + + def put[K, V](m: Map[K, V], k: K, v: V): Map[K, V] = m match { + case Tip() => singleton(k, v) + case Bin(size, k2, v2, l, r) => + genericCompare(k, k2) match { + case Less() => balance(k2, v2, put(l, k, v), r) + case Greater() => balance(k2, v2, l, put(r, k, v)) + case Equal() => Bin(size, k, v, l, r) + } + } + + + def putMax[K, V](m: Map[K, V], k: K, v: V): Map[K, V] = { + m match { + case Tip() => singleton(k, v) + case Bin(_, k2, v2, l, r) => + balance(k2, v2, l, r.putMax(k, v)) + } + } + + + def putMin[K, V](m: Map[K, V], k: K, v: V): Map[K, V] = { + m match { + case Tip() => singleton(k, v) + case Bin(_, k2, v2, l, r) => + balance(k2, v2, l.putMin(k, v), r) + } + } + + def link[K, V](k: K, v: V, l: Map[K, V], r: Map[K, V]): Map[K, V] = { + (l, r) match { + case (Tip(), r) => r.putMin(k, v) + case (l, Tip()) => l.putMax(k, v) + case (Bin(sizeL, kl, vl, ll, lr), Bin(sizeR, kr, vr, rl, rr)) => + if ((delta * sizeL) < sizeR) { balance(kr, vr, link(k, v, l, rl), rr) } + else if ((delta * sizeR) < sizeL) { balance(kl, vl, ll, link(k, v, lr, r)) } + else { bin(k, v, l, r) } + } + } + + def fromList[K, V](pairs: List[(K, V)]): Map[K, V] = { + pairs match { + case Nil() => Tip() + case Cons((k, v), Nil()) => singleton(k, v) + case Cons((k, v), rest) => + def notOrdered(k: K, pairs: List[(K, V)]) = { + pairs match { + case Nil() => false + case Cons((k2, _), _) => // k >= k2 + genericCompare(k, k2) match { + case Less() => false + case Greater() => true + case Equal() => true + } + } + } + + def insertMany(m: Map[K, V], pairs: List[(K, V)]) = { + var mapSoFar = m + pairs.foreach { case (k, v) => + mapSoFar = mapSoFar.put(k, v) + } + mapSoFar + } + + def create(level: Int, pairs: List[(K, V)]): (Map[K, V], List[(K, V)], List[(K, V)]) = { + pairs match { + case Nil() => (Tip(), [], []) + case Cons((k, v), rest) => + if (level == 1) { + val singleton = Bin(1, k, v, Tip(), Tip()) + if (notOrdered(k, rest)) { + (singleton, [], rest) + } else { + (singleton, rest, []) + } + } else { + val res = create(level.bitwiseShr(1), pairs) + res match { + case (_, Nil(), _) => res + case (l, Cons((k2, v2), Nil()), zs) => (l.putMax(k2, v2), [], zs) + case (l, Cons((k2, v2), rest2), _) => + val xs = Cons((k2, v2), rest2) // @-pattern + + if (notOrdered(k2, rest2)) { (l, [], xs) } + else { + val (r, zs, ws) = create(level.bitwiseShr(1), rest2); + (link(k2, v2, l, r), zs, ws) + } + } + } + } + } + + def go(level: Int, m: Map[K, V], pairs: List[(K, V)]): Map[K, V] = { + pairs match { + case Nil() => m + case Cons((k, v), Nil()) => m.putMax(k, v) + case Cons((k, v), rest) => + if (notOrdered(k, rest)) { insertMany(m, pairs) } + else { + val l = m; // m is the left subtree here + val cr = create(level, rest) + cr match { + case (r, xs, Nil()) => go(level.bitwiseShl(1), link(k, v, l, r), xs) + case (r, Nil(), ys) => insertMany(link(k, v, l, r), ys) + case _ => panic("create: go: cannot happen, invariant broken!") + } + } + } + } + + if (notOrdered(k, rest)) { insertMany(singleton(k, v), rest) } + else { go(1, singleton(k, v), rest) } + } + } + + def foreach[K, V](m: Map[K, V]) { action: (K, V) => Unit }: Unit = { + def go(m: Map[K, V]): Unit = { + m match { + case Tip() => () + case Bin(_, k, v, l, r) => + go(l) + action(k, v) + go(r) + } + } + go(m) + } + + def keys[K, V](m: Map[K, V]): List[K] = { + var acc = Nil() + m.foreach { (k, _v) => + acc = Cons(k, acc) + } + acc.reverse + } +} + +def collectMap[K, V, R] { stream: () => R / emit[(K, V)] }: (R, newmap::Map[K, V]) = + try { + (stream(), newmap::empty()) + } with emit[(K, V)] { case (k, v) => + val (r, map) = resume(()); + (r, map.newmap::put(k, v)) + } + +def collectMap[K, V] { stream: () => Unit / emit[(K, V)] }: newmap::Map[K, V] = + collectMap[K, V, Unit]{stream}.second + +///// === the actual problem === + +def main(): Unit = { + val m = collectMap[Int, Char] { + do emit((42, 'a')) + do emit((1, 'b')) + do emit((-6, 'c')) // comment this out to make everything work again, ... + } + val ignored = newset::fromList(m.newmap::keys) + // ... or comment this ^^^ out! + () +} \ No newline at end of file From 94e3ab95e1bfa99d1210305f9d9262d2781b5062 Mon Sep 17 00:00:00 2001 From: "effekt-updater[bot]" <181701480+effekt-updater[bot]@users.noreply.github.com> Date: Mon, 23 Dec 2024 03:34:44 +0000 Subject: [PATCH 10/31] Bump version to 0.15.0 --- package.json | 2 +- pom.xml | 2 +- project/EffektVersion.scala | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/package.json b/package.json index 33ed2f9dd..a735fe933 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "@effekt-lang/effekt", "author": "Jonathan BrachthΓ€user", - "version": "0.14.0", + "version": "0.15.0", "repository": { "type": "git", "url": "git+https://github.com/effekt-lang/effekt.git" diff --git a/pom.xml b/pom.xml index 5fccb3829..28aa9d724 100644 --- a/pom.xml +++ b/pom.xml @@ -7,7 +7,7 @@ de.bstudios Effekt Effekt - 0.14.0 + 0.15.0 4.0.0 diff --git a/project/EffektVersion.scala b/project/EffektVersion.scala index 721c68d61..95bdf67eb 100644 --- a/project/EffektVersion.scala +++ b/project/EffektVersion.scala @@ -1,4 +1,4 @@ // Don't change this file without changing the CI too! import sbt.* import sbt.Keys.* -object EffektVersion { lazy val effektVersion = "0.14.0" } +object EffektVersion { lazy val effektVersion = "0.15.0" } From 2fcfc6dfb20103fad7f7215a21ca6fc1b0354a86 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ji=C5=99=C3=AD=20Bene=C5=A1?= Date: Sat, 28 Dec 2024 15:21:49 +0100 Subject: [PATCH 11/31] CI: Prevent unnecessary release (#755) Don't release when there are no new changes. Also has a special check to not release when the only changes were in `.github/`. --- .github/workflows/autorelease.yml | 86 +++++++++++++++++++++++++++++++ 1 file changed, 86 insertions(+) diff --git a/.github/workflows/autorelease.yml b/.github/workflows/autorelease.yml index fcd9049ad..b784f1267 100644 --- a/.github/workflows/autorelease.yml +++ b/.github/workflows/autorelease.yml @@ -11,8 +11,94 @@ env: NODE_VERSION: '16.x' jobs: + check-release-needed: + name: Check if release is needed + runs-on: ubuntu-latest + steps: + - name: Checkout code + uses: actions/checkout@v4 + with: + submodules: 'true' + fetch-depth: 0 # Need full history for tags + + - name: Verify repository state + run: | + # Ensure we're on master + current_branch=$(git rev-parse --abbrev-ref HEAD) + if [ "$current_branch" != "master" ]; then + echo "ERROR: Not on master branch! Current branch: $current_branch" + exit 1 + fi + + # Ensure we have tags + if ! git tag | grep -q '^v'; then + echo "ERROR: No version tags found in repository!" + exit 1 + fi + + # Get the latest commit + latest_commit=$(git rev-parse HEAD) + echo "Latest commit: $latest_commit" + + # Get all tags on the latest commit + tags_on_commit=$(git tag --points-at $latest_commit) + echo "Tags on latest commit:" + echo "$tags_on_commit" + + # Check for version tags + if echo "$tags_on_commit" | grep -q '^v'; then + echo "Latest commit already has a version tag. Details:" + echo "$tags_on_commit" | grep '^v' + echo "No new release needed." + exit 1 + fi + + # Get the most recent version tag + previous_version=$(git tag --sort=-v:refname | grep '^v' | head -n1) + if [ -z "$previous_version" ]; then + echo "ERROR: Could not determine previous version!" + exit 1 + fi + echo "Previous version: $previous_version" + + # Check what files changed since last release + echo "Changes since $previous_version:" + changes_all=$(git diff --name-only $previous_version HEAD) + echo "$changes_all" + + # Check if only .github files changed + changes_non_github=$(echo "$changes_all" | grep -v '^\.github/') + if [ -z "$changes_non_github" ]; then + echo "Only .github files changed since $previous_version. No release needed." + exit 1 + fi + + # Verify the previous release commit message + prev_release_commit=$(git rev-list -n 1 $previous_version) + prev_commit_msg=$(git log -1 --format=%B $prev_release_commit) + prev_version_number=${previous_version#v} # Remove 'v' prefix + expected_msg="Bump version to $prev_version_number" + + if [ "$prev_commit_msg" != "$expected_msg" ]; then + echo "WARNING: Previous release commit message doesn't match expected format" + echo "Expected: $expected_msg" + echo "Found: $prev_commit_msg" + # Not failing here as this is just a warning + fi + + # Verify previous release was made by the bot + prev_release_author=$(git log -1 --format='%ae' $prev_release_commit) + if ! echo "$prev_release_author" | grep -q "effekt-updater\[bot\]@users.noreply.github.com"; then + echo "WARNING: Previous release wasn't made by effekt-updater[bot]" + echo "Author: $prev_release_author" + # Not failing here as this is just a warning + fi + + echo "All checks passed. Release is needed!" + run-tests: # redux of usual CI defined in `ci.yml` name: Run tests + needs: [check-release-needed] runs-on: ubuntu-latest steps: - name: Checkout code From 08f97ec0f25eb7b71f25d8ff2d3deed31121d16b Mon Sep 17 00:00:00 2001 From: phischu Date: Fri, 3 Jan 2025 14:46:46 +0100 Subject: [PATCH 12/31] Fix some memory leaks (#736) @marvinborner With these fixes the following does not leak anymore: ``` import array def main() = { array::allocate[Int](1) () } ``` Does this also fix other tests that are commented out? --------- Co-authored-by: Marvin Borner --- .../jvm/src/test/scala/effekt/LLVMTests.scala | 20 ++------- .../src/test/scala/effekt/StdlibTests.scala | 23 ++-------- .../effekt/generator/llvm/Transformer.scala | 3 +- .../main/scala/effekt/machine/Analysis.scala | 10 ++--- examples/stdlib/bytearray/bytearray.check | 2 +- examples/stdlib/bytearray/bytearray.effekt | 7 +++- libraries/common/effekt.effekt | 1 - libraries/common/io/filesystem.effekt | 2 + libraries/llvm/array.c | 6 ++- libraries/llvm/bytearray.c | 42 +++++++++++-------- libraries/llvm/io.c | 10 ++--- libraries/llvm/main.c | 1 - libraries/llvm/ref.c | 2 + libraries/llvm/sanity.c | 16 ------- 14 files changed, 57 insertions(+), 88 deletions(-) delete mode 100644 libraries/llvm/sanity.c diff --git a/effekt/jvm/src/test/scala/effekt/LLVMTests.scala b/effekt/jvm/src/test/scala/effekt/LLVMTests.scala index a65046cd0..356ea86cb 100644 --- a/effekt/jvm/src/test/scala/effekt/LLVMTests.scala +++ b/effekt/jvm/src/test/scala/effekt/LLVMTests.scala @@ -22,25 +22,11 @@ class LLVMTests extends EffektTests { lazy val bugs: List[File] = List( // names not sanitized (even?) examplesDir / "pos" / "special_names.effekt", - - // sanitizer/valgrind: segfault - examplesDir / "pos" / "parametrized.effekt", - - // Valgrind leak, unclear - examplesDir / "llvm" / "polymorphism_map.effekt", - examplesDir / "pos" / "type_parameters_blocks.effekt", - - // Valgrind leak in array_new - examplesDir / "benchmarks" / "are_we_fast_yet" / "sieve.effekt", - examplesDir / "benchmarks" / "are_we_fast_yet" / "nbody.effekt", - examplesDir / "benchmarks" / "are_we_fast_yet" / "bounce.effekt", // + c_ref_fresh - examplesDir / "benchmarks" / "are_we_fast_yet" / "towers.effekt", - examplesDir / "benchmarks" / "are_we_fast_yet" / "permute.effekt", - examplesDir / "benchmarks" / "are_we_fast_yet" / "queens.effekt", - examplesDir / "benchmarks" / "input_output" / "word_count_ascii.effekt", - examplesDir / "benchmarks" / "input_output" / "word_count_utf8.effekt", + // Jump to the invalid address stated on the next line examplesDir / "benchmarks" / "input_output" / "dyck_one.effekt", examplesDir / "benchmarks" / "input_output" / "number_matrix.effekt", + examplesDir / "benchmarks" / "input_output" / "word_count_ascii.effekt", + examplesDir / "benchmarks" / "input_output" / "word_count_utf8.effekt", ) /** diff --git a/effekt/jvm/src/test/scala/effekt/StdlibTests.scala b/effekt/jvm/src/test/scala/effekt/StdlibTests.scala index f74cb0e21..e3066302d 100644 --- a/effekt/jvm/src/test/scala/effekt/StdlibTests.scala +++ b/effekt/jvm/src/test/scala/effekt/StdlibTests.scala @@ -43,27 +43,10 @@ class StdlibLLVMTests extends StdlibTests { override def ignored: List[File] = List( // Toplevel let-bindings (for ANSI-color-codes in output) not supported examplesDir / "stdlib" / "test" / "test.effekt", - - // Valgrind leak/failure - examplesDir / "stdlib" / "bytearray" / "bytearray.effekt", - examplesDir / "stdlib" / "stream" / "characters.effekt", - examplesDir / "stdlib" / "stream" / "fuse_newlines.effekt", - examplesDir / "stdlib" / "io" / "filesystem" / "async_file_io.effekt", + // Syscall param write(buf) points to uninitialised byte(s) examplesDir / "stdlib" / "io" / "filesystem" / "files.effekt", + examplesDir / "stdlib" / "io" / "filesystem" / "async_file_io.effekt", + // Conditional jump or move depends on uninitialised value(s) examplesDir / "stdlib" / "io" / "filesystem" / "wordcount.effekt", - examplesDir / "stdlib" / "io" / "time.effekt", - examplesDir / "stdlib" / "list" / "flatmap.effekt", - examplesDir / "stdlib" / "list" / "zip.effekt", - examplesDir / "stdlib" / "list" / "deleteat.effekt", - examplesDir / "stdlib" / "list" / "join.effekt", - examplesDir / "stdlib" / "list" / "modifyat.effekt", - examplesDir / "stdlib" / "list" / "updateat.effekt", - examplesDir / "stdlib" / "list" / "insert.effekt", - examplesDir / "stdlib" / "list" / "fill.effekt", - examplesDir / "stdlib" / "list" / "zipwith.effekt", - examplesDir / "stdlib" / "list" / "collect.effekt", - examplesDir / "stdlib" / "list" / "build.effekt", - examplesDir / "stdlib" / "string" / "strings.effekt", - examplesDir / "stdlib" / "string" / "unicode.effekt", ) } diff --git a/effekt/shared/src/main/scala/effekt/generator/llvm/Transformer.scala b/effekt/shared/src/main/scala/effekt/generator/llvm/Transformer.scala index 5cb66a27b..476ad2329 100644 --- a/effekt/shared/src/main/scala/effekt/generator/llvm/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/generator/llvm/Transformer.scala @@ -365,11 +365,12 @@ object Transformer { eraseValues(List(v), freeVariables(rest)); transform(rest) - case machine.ForeignCall(machine.Variable(resultName, resultType), foreign, values, rest) => + case machine.ForeignCall(variable @ machine.Variable(resultName, resultType), foreign, values, rest) => emit(Comment(s"foreignCall $resultName : $resultType, foreign $foreign, ${values.length} values")) val functionType = PointerType(); shareValues(values, freeVariables(rest)); emit(Call(resultName, Ccc(), transform(resultType), ConstantGlobal(foreign), values.map(transform))); + eraseValues(List(variable), freeVariables(rest)) transform(rest) case machine.Statement.Hole => diff --git a/effekt/shared/src/main/scala/effekt/machine/Analysis.scala b/effekt/shared/src/main/scala/effekt/machine/Analysis.scala index d61cd77b5..06bb92ce0 100644 --- a/effekt/shared/src/main/scala/effekt/machine/Analysis.scala +++ b/effekt/shared/src/main/scala/effekt/machine/Analysis.scala @@ -20,7 +20,7 @@ def freeVariables(statement: Statement): Set[Variable] = case Jump(Label(_, environment)) => environment.toSet case Substitute(bindings, rest) => - freeVariables(rest) -- bindings.map(_._1).toSet ++ bindings.map(_._2).toSet + freeVariables(rest).map { variable => bindings.toMap.getOrElse(variable, variable) } case Construct(name, tag, values, rest) => Set.from(values) ++ (freeVariables(rest) -- Set(name)) case Switch(value, clauses, default: Option[Clause]) => @@ -30,9 +30,9 @@ def freeVariables(statement: Statement): Set[Variable] = case Invoke(value, tag, values) => Set(value) ++ Set.from(values) case Var(name, init, tpe, rest) => - freeVariables(rest) ++ Set(init) -- Set(name) + Set(init) ++ (freeVariables(rest) -- Set(name)) case LoadVar(name, ref, rest) => - Set(ref) ++ freeVariables(rest) -- Set(name) + Set(ref) ++ (freeVariables(rest) -- Set(name)) case StoreVar(ref, value, rest) => Set(ref, value) ++ freeVariables(rest) case PushFrame(frame, rest) => @@ -44,7 +44,7 @@ def freeVariables(statement: Statement): Set[Variable] = case Resume(value, rest) => Set(value) ++ freeVariables(rest) case Shift(name, prompt, rest) => - freeVariables(rest) -- Set(name) ++ Set(prompt) + Set(prompt) ++ (freeVariables(rest) -- Set(name)) case LiteralInt(name, value, rest) => freeVariables(rest) - name case LiteralDouble(name, value, rest) => @@ -52,6 +52,6 @@ def freeVariables(statement: Statement): Set[Variable] = case LiteralUTF8String(name, utf8, rest) => freeVariables(rest) - name case ForeignCall(name, builtin, arguments, rest) => - arguments.toSet ++ freeVariables(rest) - name + arguments.toSet ++ (freeVariables(rest) - name) case Hole => Set.empty } diff --git a/examples/stdlib/bytearray/bytearray.check b/examples/stdlib/bytearray/bytearray.check index 1257dbb58..7c50b1575 100644 --- a/examples/stdlib/bytearray/bytearray.check +++ b/examples/stdlib/bytearray/bytearray.check @@ -1 +1 @@ -hfllo +hfllo!!! diff --git a/examples/stdlib/bytearray/bytearray.effekt b/examples/stdlib/bytearray/bytearray.effekt index dfa719b8d..1bcac7ebf 100644 --- a/examples/stdlib/bytearray/bytearray.effekt +++ b/examples/stdlib/bytearray/bytearray.effekt @@ -2,14 +2,17 @@ import bytearray def main() = { - val b = bytearray::allocate(16); + val b = bytearray::allocate(8); b.unsafeSet(0, 104.toByte) b.unsafeSet(1, 101.toByte) b.unsafeSet(2, 108.toByte) b.unsafeSet(3, 108.toByte) b.unsafeSet(4, 111.toByte) + b.unsafeSet(5, 33.toByte) + b.unsafeSet(6, 33.toByte) + b.unsafeSet(7, 33.toByte) b.unsafeSet(1, 102.toByte) - println(b.toString) // hfllo + println(b.toString) // hfllo!!! } diff --git a/libraries/common/effekt.effekt b/libraries/common/effekt.effekt index 5fb10bea9..09017e5ac 100644 --- a/libraries/common/effekt.effekt +++ b/libraries/common/effekt.effekt @@ -115,7 +115,6 @@ extern pure def length(str: String): Int = chez "(string-length ${str})" llvm """ %x = call %Int @c_bytearray_size(%Pos ${str}) - call void @erasePositive(%Pos ${str}) ret %Int %x """ diff --git a/libraries/common/io/filesystem.effekt b/libraries/common/io/filesystem.effekt index 1dad752c9..6ec241472 100644 --- a/libraries/common/io/filesystem.effekt +++ b/libraries/common/io/filesystem.effekt @@ -198,6 +198,7 @@ namespace internal { ret void """ + /// The buffer must be kept alive by using it after the call extern async def read(file: Int, buffer: ByteArray, offset: Int, size: Int, position: Int): Int = jsNode "$effekt.capture(callback => read(${file}, ${buffer}, ${offset}, ${size}, ${position}, callback))" llvm """ @@ -205,6 +206,7 @@ namespace internal { ret void """ + /// The buffer must be kept alive by using it after the call extern async def write(file: Int, buffer: ByteArray, offset: Int, size: Int, position: Int): Int = jsNode "$effekt.capture(callback => write(${file}, ${buffer}, ${offset}, ${size}, ${position}, callback))" llvm """ diff --git a/libraries/llvm/array.c b/libraries/llvm/array.c index adb2a1f08..df0950b82 100644 --- a/libraries/llvm/array.c +++ b/libraries/llvm/array.c @@ -32,13 +32,16 @@ struct Pos c_array_new(const Int size) { Int c_array_size(const struct Pos arr) { uint64_t *sizePtr = arr.obj + sizeof(struct Header); - return *sizePtr; + uint64_t size = *sizePtr; + erasePositive(arr); + return size; } struct Pos c_array_get(const struct Pos arr, const Int index) { struct Pos *dataPtr = arr.obj + sizeof(struct Header) + sizeof(uint64_t); struct Pos element = dataPtr[index]; sharePositive(element); + erasePositive(arr); return element; } @@ -47,6 +50,7 @@ struct Pos c_array_set(const struct Pos arr, const Int index, const struct Pos v struct Pos element = dataPtr[index]; erasePositive(element); dataPtr[index] = value; + erasePositive(arr); return Unit; } diff --git a/libraries/llvm/bytearray.c b/libraries/llvm/bytearray.c index 1866602ea..7cf5a3ade 100644 --- a/libraries/llvm/bytearray.c +++ b/libraries/llvm/bytearray.c @@ -29,30 +29,34 @@ struct Pos c_bytearray_new(const Int size) { } Int c_bytearray_size(const struct Pos arr) { + erasePositive(arr); return arr.tag; } -uint8_t* c_bytearray_data(const struct Pos arr) { - return arr.obj + sizeof(struct Header); -} - Byte c_bytearray_get(const struct Pos arr, const Int index) { Byte *dataPtr = arr.obj + sizeof(struct Header); Byte element = dataPtr[index]; + erasePositive(arr); return element; } struct Pos c_bytearray_set(const struct Pos arr, const Int index, const Byte value) { Byte *dataPtr = arr.obj + sizeof(struct Header); dataPtr[index] = value; + erasePositive(arr); return Unit; } +// Internal Operations + +uint8_t* c_bytearray_data(const struct Pos arr) { + uint8_t *data = arr.obj + sizeof(struct Header); + return data; +} + struct Pos c_bytearray_construct(const uint64_t n, const uint8_t *data) { struct Pos arr = c_bytearray_new(n); - memcpy(c_bytearray_data(arr), data, n); - return arr; } @@ -66,7 +70,7 @@ struct Pos c_bytearray_from_nullterminated_string(const char *data) { } char* c_bytearray_into_nullterminated_string(const struct Pos arr) { - uint64_t size = c_bytearray_size(arr); + uint64_t size = arr.tag; char* result = (char*)malloc(size + 1); @@ -125,12 +129,14 @@ struct Pos c_bytearray_show_Double(const Double x) { // TODO do this in Effekt struct Pos c_bytearray_concatenate(const struct Pos left, const struct Pos right) { - const struct Pos concatenated = c_bytearray_new(c_bytearray_size(left) + c_bytearray_size(right)); - for (int64_t j = 0; j < c_bytearray_size(concatenated); ++j) + uint64_t left_size = left.tag; + uint64_t right_size = right.tag; + const struct Pos concatenated = c_bytearray_new(left_size + right_size); + for (uint64_t j = 0; j < concatenated.tag; ++j) c_bytearray_data(concatenated)[j] - = j < c_bytearray_size(left) + = j < left_size ? c_bytearray_data(left)[j] - : c_bytearray_data(right)[j - c_bytearray_size(left)]; + : c_bytearray_data(right)[j - left_size]; erasePositive(left); erasePositive(right); @@ -139,8 +145,8 @@ struct Pos c_bytearray_concatenate(const struct Pos left, const struct Pos right // TODO do this in Effekt struct Pos c_bytearray_equal(const struct Pos left, const struct Pos right) { - uint64_t left_size = c_bytearray_size(left); - uint64_t right_size = c_bytearray_size(right); + uint64_t left_size = left.tag; + uint64_t right_size = right.tag; if (left_size != right_size) { erasePositive(left); erasePositive(right); @@ -161,7 +167,7 @@ struct Pos c_bytearray_equal(const struct Pos left, const struct Pos right) { // TODO deprecate struct Pos c_bytearray_substring(const struct Pos str, uint64_t start, uint64_t end) { const struct Pos substr = c_bytearray_new(end - start); - for (int64_t j = 0; j < c_bytearray_size(substr); ++j) { + for (uint64_t j = 0; j < substr.tag; ++j) { c_bytearray_data(substr)[j] = c_bytearray_data(str)[start+j]; } erasePositive(str); @@ -174,27 +180,27 @@ uint32_t c_bytearray_character_at(const struct Pos str, const uint64_t index) { uint8_t first_byte = bytes[index]; uint32_t character = 0; - uint32_t length = c_bytearray_size(str); + uint32_t size = str.tag; if (first_byte < 0x80) { // Single-byte character (0xxxxxxx) character = first_byte; } else if ((first_byte & 0xE0) == 0xC0) { // Two-byte character (110xxxxx 10xxxxxx) - if (index + 1 < length) { + if (index + 1 < size) { character = ((first_byte & 0x1F) << 6) | (bytes[index + 1] & 0x3F); } } else if ((first_byte & 0xF0) == 0xE0) { // Three-byte character (1110xxxx 10xxxxxx 10xxxxxx) - if (index + 2 < length) { + if (index + 2 < size) { character = ((first_byte & 0x0F) << 12) | ((bytes[index + 1] & 0x3F) << 6) | (bytes[index + 2] & 0x3F); } } else if ((first_byte & 0xF8) == 0xF0) { // Four-byte character (11110xxx 10xxxxxx 10xxxxxx 10xxxxxx) - if (index + 3 < length) { + if (index + 3 < size) { character = ((first_byte & 0x07) << 18) | ((bytes[index + 1] & 0x3F) << 12) | ((bytes[index + 2] & 0x3F) << 6) | diff --git a/libraries/llvm/io.c b/libraries/llvm/io.c index cff63fc0b..bdc3cfbbb 100644 --- a/libraries/llvm/io.c +++ b/libraries/llvm/io.c @@ -20,7 +20,7 @@ void c_io_println_Double(const Double x) { } void c_io_println_String(String text) { - for (int64_t j = 0; j < c_bytearray_size(text); ++j) + for (uint64_t j = 0; j < text.tag; ++j) putchar(c_bytearray_data(text)[j]); erasePositive(text); putchar('\n'); @@ -126,8 +126,8 @@ void c_fs_open(String path, struct Pos mode, Stack stack) { void c_fs_read(Int file, struct Pos buffer, Int offset, Int size, Int position, Stack stack) { uv_buf_t buf = uv_buf_init((char*)(c_bytearray_data(buffer) + offset), size); - // erasePositive(buffer); - // TODO we should erase the buffer but abort if this was the last reference + erasePositive(buffer); + // TODO panic if this was the last reference uv_fs_t* request = malloc(sizeof(uv_fs_t)); request->data = stack; @@ -144,8 +144,8 @@ void c_fs_read(Int file, struct Pos buffer, Int offset, Int size, Int position, void c_fs_write(Int file, struct Pos buffer, Int offset, Int size, Int position, Stack stack) { uv_buf_t buf = uv_buf_init((char*)(c_bytearray_data(buffer) + offset), size); - // erasePositive(buffer); - // TODO we should erase the buffer but abort if this was the last reference + erasePositive(buffer); + // TODO panic if this was the last reference uv_fs_t* request = malloc(sizeof(uv_fs_t)); request->data = stack; diff --git a/libraries/llvm/main.c b/libraries/llvm/main.c index f11320541..34e22d5c9 100644 --- a/libraries/llvm/main.c +++ b/libraries/llvm/main.c @@ -9,7 +9,6 @@ #define DEBUG_REFCOUNT (false) -#include "sanity.c" #include "types.c" #include "bytearray.c" #include "io.c" diff --git a/libraries/llvm/ref.c b/libraries/llvm/ref.c index aa40032fc..48620a42c 100644 --- a/libraries/llvm/ref.c +++ b/libraries/llvm/ref.c @@ -31,6 +31,7 @@ struct Pos c_ref_get(const struct Pos ref) { struct Pos *fieldPtr = ref.obj + sizeof(struct Header); struct Pos element = *fieldPtr; sharePositive(element); + erasePositive(ref); return element; } @@ -39,6 +40,7 @@ struct Pos c_ref_set(const struct Pos ref, const struct Pos value) { struct Pos element = *fieldPtr; erasePositive(element); *fieldPtr = value; + erasePositive(ref); return Unit; } diff --git a/libraries/llvm/sanity.c b/libraries/llvm/sanity.c deleted file mode 100644 index a64c8f564..000000000 --- a/libraries/llvm/sanity.c +++ /dev/null @@ -1,16 +0,0 @@ -#ifndef EFFEKT_SANITY_C -#define EFFEKT_SANITY_C - - -#if CHAR_BIT != 8 -#error Machine ought to have eight bits in a byte. -#endif - - -#define ASSERT_NON_NULL(PTR) if ((PTR) == NULL) { \ - fprintf(stderr, "*** MALLOC PANIC\n"); \ - fflush(stderr); \ - exit(1); } - - -#endif From 40c0706b06a4bc2c013f1d17901d6ce52acb9afc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jonathan=20Immanuel=20Brachth=C3=A4user?= Date: Sat, 4 Jan 2025 12:03:45 +0100 Subject: [PATCH 13/31] Rework Optimizer (#695) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: dvdvgt <40773635+dvdvgt@users.noreply.github.com> Co-authored-by: effekt-updater[bot] <181701480+effekt-updater[bot]@users.noreply.github.com> Co-authored-by: JiΕ™Γ­ BeneΕ‘ Co-authored-by: Philipp Schuster Co-authored-by: Serkan Muhcu Co-authored-by: Marvin Co-authored-by: Jakub <168542356+JakubSchwenkbeck@users.noreply.github.com> Co-authored-by: Marcial Gaißert Co-authored-by: Can <32599057+CanCodes@users.noreply.github.com> --- .../src/test/scala/effekt/StdlibTests.scala | 10 +- .../scala/effekt/core/OptimizerTests.scala | 38 +- .../src/main/scala/effekt/core/Inline.scala | 284 ------------- .../scala/effekt/core/LambdaLifting.scala | 4 +- .../main/scala/effekt/core/Optimizer.scala | 36 -- .../effekt/core/PolymorphismBoxing.scala | 2 +- .../scala/effekt/core/PrettyPrinter.scala | 8 +- .../src/main/scala/effekt/core/Renamer.scala | 11 +- .../src/main/scala/effekt/core/Tree.scala | 228 ++++------ .../core/optimizer/BindSubexpressions.scala | 199 +++++++++ .../core/{ => optimizer}/Deadcode.scala | 21 +- .../effekt/core/optimizer/DropBindings.scala | 66 +++ .../effekt/core/optimizer/Normalizer.scala | 402 ++++++++++++++++++ .../effekt/core/optimizer/Optimizer.scala | 54 +++ .../core/{ => optimizer}/Reachable.scala | 69 +-- .../optimizer/RemoveTailResumptions.scala | 73 ++++ .../{ => optimizer}/StaticArguments.scala | 58 +-- .../effekt/generator/chez/ChezScheme.scala | 5 +- .../effekt/generator/chez/Transformer.scala | 4 +- .../effekt/generator/js/JavaScript.scala | 3 +- .../scala/effekt/generator/llvm/LLVM.scala | 7 +- .../scala/effekt/machine/Transformer.scala | 45 +- .../src/main/scala/effekt/util/Debug.scala | 4 +- .../benchmarks/are_we_fast_yet/queens.effekt | 1 - .../parsing_dollars.effekt | 2 +- .../effect_handlers_bench/tree_explore.effekt | 1 - libraries/common/io.effekt | 3 +- libraries/llvm/rts.ll | 20 + 28 files changed, 1059 insertions(+), 599 deletions(-) delete mode 100644 effekt/shared/src/main/scala/effekt/core/Inline.scala delete mode 100644 effekt/shared/src/main/scala/effekt/core/Optimizer.scala create mode 100644 effekt/shared/src/main/scala/effekt/core/optimizer/BindSubexpressions.scala rename effekt/shared/src/main/scala/effekt/core/{ => optimizer}/Deadcode.scala (68%) create mode 100644 effekt/shared/src/main/scala/effekt/core/optimizer/DropBindings.scala create mode 100644 effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala create mode 100644 effekt/shared/src/main/scala/effekt/core/optimizer/Optimizer.scala rename effekt/shared/src/main/scala/effekt/core/{ => optimizer}/Reachable.scala (70%) create mode 100644 effekt/shared/src/main/scala/effekt/core/optimizer/RemoveTailResumptions.scala rename effekt/shared/src/main/scala/effekt/core/{ => optimizer}/StaticArguments.scala (80%) diff --git a/effekt/jvm/src/test/scala/effekt/StdlibTests.scala b/effekt/jvm/src/test/scala/effekt/StdlibTests.scala index e3066302d..d362b2384 100644 --- a/effekt/jvm/src/test/scala/effekt/StdlibTests.scala +++ b/effekt/jvm/src/test/scala/effekt/StdlibTests.scala @@ -41,11 +41,17 @@ class StdlibLLVMTests extends StdlibTests { override def debug = sys.env.get("EFFEKT_DEBUG").nonEmpty override def ignored: List[File] = List( - // Toplevel let-bindings (for ANSI-color-codes in output) not supported - examplesDir / "stdlib" / "test" / "test.effekt", + // segfaults + examplesDir / "stdlib" / "stream" / "fuse_newlines.effekt", + + // valgrind + examplesDir / "stdlib" / "list" / "modifyat.effekt", + examplesDir / "stdlib" / "list" / "updateat.effekt", + // Syscall param write(buf) points to uninitialised byte(s) examplesDir / "stdlib" / "io" / "filesystem" / "files.effekt", examplesDir / "stdlib" / "io" / "filesystem" / "async_file_io.effekt", + // Conditional jump or move depends on uninitialised value(s) examplesDir / "stdlib" / "io" / "filesystem" / "wordcount.effekt", ) diff --git a/effekt/jvm/src/test/scala/effekt/core/OptimizerTests.scala b/effekt/jvm/src/test/scala/effekt/core/OptimizerTests.scala index c6d933961..dc91897d9 100644 --- a/effekt/jvm/src/test/scala/effekt/core/OptimizerTests.scala +++ b/effekt/jvm/src/test/scala/effekt/core/OptimizerTests.scala @@ -1,5 +1,8 @@ package effekt package core + +import effekt.core.optimizer.* + import effekt.symbols class OptimizerTests extends CoreTests { @@ -30,15 +33,11 @@ class OptimizerTests extends CoreTests { Deadcode.remove(Set(mainSymbol), tree) } - def inlineOnce(input: String, expected: String)(using munit.Location) = - assertTransformsTo(input, expected) { tree => - val (result, count) = Inline.once(Set(mainSymbol), tree, 50) - result - } - - def inlineFull(input: String, expected: String)(using munit.Location) = + def normalize(input: String, expected: String)(using munit.Location) = assertTransformsTo(input, expected) { tree => - Inline.full(Set(mainSymbol), tree, 50) + val anfed = BindSubexpressions.transform(tree) + val normalized = Normalizer.normalize(Set(mainSymbol), anfed, 50) + Deadcode.remove(mainSymbol, normalized) } test("toplevel"){ @@ -155,11 +154,10 @@ class OptimizerTests extends CoreTests { |""".stripMargin val expected = - """ def foo = { () => return 42 } - | def main = { () => return 42 } + """ def main = { () => return 42 } |""".stripMargin - inlineOnce(input, expected) + normalize(input, expected) } test("inline with argument"){ @@ -169,11 +167,10 @@ class OptimizerTests extends CoreTests { |""".stripMargin val expected = - """ def foo = { (n: Int) => return n:Int } - | def main = { () => return 42 } + """ def main = { () => return 42 } |""".stripMargin - inlineOnce(input, expected) + normalize(input, expected) } test("inline higher order function"){ @@ -188,17 +185,10 @@ class OptimizerTests extends CoreTests { |""".stripMargin val expected = - """ def foo = { (n: Int) => return n:Int } - | def hof = { (){f : (Int) => Int} => - | (f : (Int) => Int @ {f})(1) - | } - | def main = { () => - | def local(n: Int) = return n:Int - | (local : (Int) => Int @ {})(1) - | } + """ def main = { () => return 1 } |""".stripMargin - inlineOnce(input, expected) + normalize(input, expected) } test("fully inline higher order function"){ @@ -216,7 +206,7 @@ class OptimizerTests extends CoreTests { """ def main = { () => return 1 } |""".stripMargin - inlineFull(input, expected) + normalize(input, expected) } } diff --git a/effekt/shared/src/main/scala/effekt/core/Inline.scala b/effekt/shared/src/main/scala/effekt/core/Inline.scala deleted file mode 100644 index 8fc01284e..000000000 --- a/effekt/shared/src/main/scala/effekt/core/Inline.scala +++ /dev/null @@ -1,284 +0,0 @@ -package effekt -package core - -import effekt.core.Block.BlockLit -import effekt.core.Pure.ValueVar -import effekt.core.normal.* -import effekt.util.messages.INTERNAL_ERROR - -import scala.collection.mutable -import kiama.util.Counter - -/** - * Inlines block definitions. - * - * 1. First computes usage (using [[Reachable.apply]]) - * 2. Top down traversal where we inline definitions - * - * Invariants: - * - the context `defs` always contains the _original_ definitions, not rewritten ones. - * Rewriting them has to be performed at the inline-site. - */ -object Inline { - - case class InlineContext( - // is mutable to update when introducing temporaries; - // they should also be visible after leaving a scope (so mutable.Map and not `var usage`). - usage: mutable.Map[Id, Usage], - defs: Map[Id, Definition], - maxInlineSize: Int, - inlineCount: Counter = Counter(0) - ) { - def ++(other: Map[Id, Definition]): InlineContext = InlineContext(usage, defs ++ other, maxInlineSize, inlineCount) - - def ++(other: List[Definition]): InlineContext = ++(other.map(d => d.id -> d).toMap) - - def ++=(fresh: Map[Id, Usage]): Unit = { usage ++= fresh } - } - - def once(entrypoints: Set[Id], m: ModuleDecl, maxInlineSize: Int): (ModuleDecl, Int) = { - val usage = Reachable(m) ++ entrypoints.map(id => id -> Usage.Many).toMap - val defs = m.definitions.map(d => d.id -> d).toMap - val context = InlineContext(mutable.Map.from(usage), defs, maxInlineSize) - - val (updatedDefs, _) = rewrite(m.definitions)(using context) - (m.copy(definitions = updatedDefs), context.inlineCount.value) - } - - def full(entrypoints: Set[Id], m: ModuleDecl, maxInlineSize: Int): ModuleDecl = - var lastCount = 1 - var tree = m - while (lastCount > 0) { - val (inlined, count) = Inline.once(entrypoints, tree, maxInlineSize) - // (3) drop unused definitions after inlining - tree = Deadcode.remove(entrypoints, inlined) - lastCount = count - } - tree - - def shouldInline(id: Id)(using ctx: InlineContext): Boolean = - ctx.usage.get(id) match { - case None => false - case Some(Usage.Once) => true - case Some(Usage.Recursive) => false // we don't inline recursive functions for the moment - case Some(Usage.Many) => - ctx.defs.get(id).exists { d => - def isSmall = d.size <= ctx.maxInlineSize - def isHigherOrder = d match { - case Definition.Def(id, BlockLit(_, _, _, bparams, _)) => - bparams.exists(p => p.tpe match { - case t: BlockType.Function => true - case t: BlockType.Interface => false - }) - case _ => false - } - isSmall || isHigherOrder - } - } - - def shouldKeep(id: Id)(using ctx: InlineContext): Boolean = - ctx.usage.get(id) match { - case None => false - case Some(Usage.Once) => false - case Some(Usage.Recursive) => true // we don't inline recursive functions for the moment - case Some(Usage.Many) => true - } - - def used(id: Id)(using ctx: InlineContext): Boolean = - ctx.usage.isDefinedAt(id) - - /** - * Rewrites the list of definition and returns: - * 1. the updated list - * 2. definitions - * a. original defnitions: in case we need to dealias elsewhere - * b. the updated definitions, where the rhs might have been dealiased already (see #733) - */ - def rewrite(definitions: List[Definition])(using ctx: InlineContext): (List[Definition], InlineContext) = - given allDefs: InlineContext = ctx ++ definitions - - val filtered = definitions.collect { - case Definition.Def(id, block) => Definition.Def(id, rewrite(block)) - // we drop aliases - case Definition.Let(id, tpe, binding) if !binding.isInstanceOf[ValueVar] => - Definition.Let(id, tpe, rewrite(binding)) - } - (filtered, allDefs ++ filtered) - - def blockDefFor(id: Id)(using ctx: InlineContext): Option[Block] = - ctx.defs.get(id) map { - case Definition.Def(id, block) => block - case Definition.Let(id, _, binding) => INTERNAL_ERROR("Should not happen") - } - - def dealias(b: Block.BlockVar)(using ctx: InlineContext): BlockVar = - ctx.defs.get(b.id) match { - case Some(Definition.Def(id, aliased : Block.BlockVar)) => dealias(aliased) - case _ => b - } - - def dealias(b: Pure.ValueVar)(using ctx: InlineContext): ValueVar = - ctx.defs.get(b.id) match { - case Some(Definition.Let(id, _, aliased : Pure.ValueVar)) => dealias(aliased) - case _ => b - } - - def rewrite(d: Definition)(using InlineContext): Definition = d match { - case Definition.Def(id, block) => Definition.Def(id, rewrite(block)) - case Definition.Let(id, tpe, binding) => Definition.Let(id, tpe, rewrite(binding)) - } - - def rewrite(s: Stmt)(using C: InlineContext): Stmt = s match { - case Stmt.Scope(definitions, body) => - val (defs, ctx) = rewrite(definitions) - scope(defs, rewrite(body)(using ctx)) - - case Stmt.App(b, targs, vargs, bargs) => - app(rewrite(b), targs, vargs.map(rewrite), bargs.map(rewrite)) - - case Stmt.Invoke(b, method, methodTpe, targs, vargs, bargs) => - invoke(rewrite(b), method, methodTpe, targs, vargs.map(rewrite), bargs.map(rewrite)) - - case Stmt.Reset(body) => - rewrite(body) match { - case BlockLit(tparams, cparams, vparams, List(prompt), body) if !used(prompt.id) => body - case b => Stmt.Reset(b) - } - - // congruences - case Stmt.Return(expr) => Return(rewrite(expr)) - case Stmt.Val(id, tpe, binding, body) => valDef(id, tpe, rewrite(binding), rewrite(body)) - case Stmt.If(cond, thn, els) => If(rewrite(cond), rewrite(thn), rewrite(els)) - case Stmt.Match(scrutinee, clauses, default) => - patternMatch(rewrite(scrutinee), clauses.map { case (id, value) => id -> rewrite(value) }, default.map(rewrite)) - case Stmt.Alloc(id, init, region, body) => Alloc(id, rewrite(init), region, rewrite(body)) - case Stmt.Shift(prompt, b @ BlockLit(tparams, cparams, vparams, List(k), body)) if tailResumptive(k.id, body) => - C.inlineCount.next() - rewrite(removeTailResumption(k.id, body)) - - case Stmt.Shift(prompt, body) => Shift(prompt, rewrite(body)) - - - case Stmt.Resume(k, body) => Resume(k, rewrite(body)) - case Stmt.Region(body) => Region(rewrite(body)) - case Stmt.Var(id, init, capture, body) => Stmt.Var(id, rewrite(init), capture, rewrite(body)) - case Stmt.Get(id, capt, tpe) => Stmt.Get(id, capt, tpe) - case Stmt.Put(id, capt, value) => Stmt.Put(id, capt, rewrite(value)) - case Stmt.Hole() => s - } - def rewrite(b: BlockLit)(using InlineContext): BlockLit = - b match { - case BlockLit(tparams, cparams, vparams, bparams, body) => - BlockLit(tparams, cparams, vparams, bparams, rewrite(body)) - } - - def rewrite(b: Block)(using C: InlineContext): Block = b match { - case Block.BlockVar(id, _, _) if shouldInline(id) => - blockDefFor(id) match { - case Some(value) => - C.inlineCount.next() - Renamer.rename(value) - case None => b - } - case b @ Block.BlockVar(id, _, _) => dealias(b) - - // congruences - case b @ Block.BlockLit(tparams, cparams, vparams, bparams, body) => rewrite(b) - case Block.Unbox(pure) => unbox(rewrite(pure)) - case Block.New(impl) => New(rewrite(impl)) - } - - def rewrite(s: Implementation)(using InlineContext): Implementation = - s match { - case Implementation(interface, operations) => Implementation(interface, operations.map { op => - op.copy(body = rewrite(op.body)) - }) - } - - def rewrite(p: Pure)(using InlineContext): Pure = p match { - case Pure.PureApp(b, targs, vargs) => pureApp(rewrite(b), targs, vargs.map(rewrite)) - case Pure.Make(data, tag, vargs) => make(data, tag, vargs.map(rewrite)) - // currently, we don't inline values, but we can dealias them - case x @ Pure.ValueVar(id, annotatedType) => dealias(x) - - // congruences - case Pure.Literal(value, annotatedType) => p - case Pure.Select(target, field, annotatedType) => select(rewrite(target), field, annotatedType) - case Pure.Box(b, annotatedCapture) => box(rewrite(b), annotatedCapture) - } - - def rewrite(e: Expr)(using InlineContext): Expr = e match { - case DirectApp(b, targs, vargs, bargs) => directApp(rewrite(b), targs, vargs.map(rewrite), bargs.map(rewrite)) - - // congruences - case Run(s) => run(rewrite(s)) - case pure: Pure => rewrite(pure) - } - - case class Binding[A](run: (A => Stmt) => Stmt) { - def flatMap[B](rest: A => Binding[B]): Binding[B] = { - Binding(k => run(a => rest(a).run(k))) - } - } - - def pure[A](a: A): Binding[A] = Binding(k => k(a)) - - // A simple syntactic check whether this stmt is tailresumptive in k - def tailResumptive(k: Id, stmt: Stmt): Boolean = - def freeInStmt(stmt: Stmt): Boolean = Variables.free(stmt).containsBlock(k) - def freeInExpr(expr: Expr): Boolean = Variables.free(expr).containsBlock(k) - def freeInDef(definition: Definition): Boolean = Variables.free(definition).containsBlock(k) - - stmt match { - case Stmt.Scope(definitions, body) => definitions.forall { d => !freeInDef(d) } && tailResumptive(k, body) - case Stmt.Return(expr) => false - case Stmt.Val(id, annotatedTpe, binding, body) => tailResumptive(k, body) && !freeInStmt(binding) - case Stmt.App(callee, targs, vargs, bargs) => false - case Stmt.Invoke(callee, method, methodTpe, targs, vargs, bargs) => false - case Stmt.If(cond, thn, els) => !freeInExpr(cond) && tailResumptive(k, thn) && tailResumptive(k, els) - // Interestingly, we introduce a join point making this more difficult to implement properly - case Stmt.Match(scrutinee, clauses, default) => !freeInExpr(scrutinee) && clauses.forall { - case (_, BlockLit(tparams, cparams, vparams, bparams, body)) => tailResumptive(k, body) - } && default.forall { stmt => tailResumptive(k, stmt) } - case Stmt.Region(BlockLit(tparams, cparams, vparams, bparams, body)) => tailResumptive(k, body) - case Stmt.Region(_) => ??? - case Stmt.Alloc(id, init, region, body) => tailResumptive(k, body) && !freeInExpr(init) - case Stmt.Var(id, init, capture, body) => tailResumptive(k, body) && !freeInExpr(init) - case Stmt.Get(id, annotatedCapt, annotatedTpe) => false - case Stmt.Put(id, annotatedCapt, value) => false - case Stmt.Reset(BlockLit(tparams, cparams, vparams, bparams, body)) => tailResumptive(k, body) // is this correct? - case Stmt.Shift(prompt, body) => false - case Stmt.Resume(k2, body) => k2.id == k // what if k is free in body? - case Stmt.Hole() => true - } - - def removeTailResumption(k: Id, stmt: Stmt): Stmt = stmt match { - case Stmt.Scope(definitions, body) => Stmt.Scope(definitions, removeTailResumption(k, body)) - case Stmt.Val(id, annotatedTpe, binding, body) => Stmt.Val(id, annotatedTpe, binding, removeTailResumption(k, body)) - case Stmt.If(cond, thn, els) => Stmt.If(cond, removeTailResumption(k, thn), removeTailResumption(k, els)) - case Stmt.Match(scrutinee, clauses, default) => Stmt.Match(scrutinee, clauses.map { - case (tag, block) => tag -> removeTailResumption(k, block) - }, default.map(removeTailResumption(k, _))) - case Stmt.Region(body : BlockLit) => - Stmt.Region(removeTailResumption(k, body)) - case Stmt.Region(_) => ??? - case Stmt.Alloc(id, init, region, body) => Stmt.Alloc(id, init, region, removeTailResumption(k, body)) - case Stmt.Var(id, init, capture, body) => Stmt.Var(id, init, capture, removeTailResumption(k, body)) - case Stmt.Reset(body) => Stmt.Reset(removeTailResumption(k, body)) - case Stmt.Resume(k2, body) if k2.id == k => body - - case Stmt.Resume(k, body) => stmt - case Stmt.Shift(prompt, body) => stmt - case Stmt.Hole() => stmt - case Stmt.Return(expr) => stmt - case Stmt.App(callee, targs, vargs, bargs) => stmt - case Stmt.Invoke(callee, method, methodTpe, targs, vargs, bargs) => stmt - case Stmt.Get(id, annotatedCapt, annotatedTpe) => stmt - case Stmt.Put(id, annotatedCapt, value) => stmt - } - - def removeTailResumption(k: Id, block: BlockLit): BlockLit = block match { - case BlockLit(tparams, cparams, vparams, bparams, body) => - BlockLit(tparams, cparams, vparams, bparams, removeTailResumption(k, body)) - } -} diff --git a/effekt/shared/src/main/scala/effekt/core/LambdaLifting.scala b/effekt/shared/src/main/scala/effekt/core/LambdaLifting.scala index 28fcee820..27afab523 100644 --- a/effekt/shared/src/main/scala/effekt/core/LambdaLifting.scala +++ b/effekt/shared/src/main/scala/effekt/core/LambdaLifting.scala @@ -7,8 +7,6 @@ import scala.collection.mutable import effekt.core.Variables import effekt.core.Variables.{ all, bound, free } -import effekt.core.normal.scope - class LambdaLifting(m: core.ModuleDecl)(using Context) extends core.Tree.Rewrite { val locals = Locals(m) @@ -61,7 +59,7 @@ class LambdaLifting(m: core.ModuleDecl)(using Context) extends core.Tree.Rewrite override def stmt = { case core.Scope(defs, body) => - scope(defs.flatMap { + MaybeScope(defs.flatMap { // we lift named local definitions to the toplevel case Definition.Def(id, BlockLit(tparams, cparams, vparams, bparams, body)) => lifted.append(Definition.Def(id, diff --git a/effekt/shared/src/main/scala/effekt/core/Optimizer.scala b/effekt/shared/src/main/scala/effekt/core/Optimizer.scala deleted file mode 100644 index a1dd8f80b..000000000 --- a/effekt/shared/src/main/scala/effekt/core/Optimizer.scala +++ /dev/null @@ -1,36 +0,0 @@ -package effekt -package core - -import effekt.PhaseResult.CoreTransformed -import effekt.context.Context - -import kiama.util.Source - -object Optimizer extends Phase[CoreTransformed, CoreTransformed] { - - val phaseName: String = "core-optimizer" - - def run(input: CoreTransformed)(using Context): Option[CoreTransformed] = - input match { - case CoreTransformed(source, tree, mod, core) => - val term = Context.checkMain(mod) - val optimized = optimize(source, term, core) - Some(CoreTransformed(source, tree, mod, optimized)) - } - - def optimize(source: Source, mainSymbol: symbols.Symbol, core: ModuleDecl)(using Context): ModuleDecl = - // (1) first thing we do is simply remove unused definitions (this speeds up all following analysis and rewrites) - val tree = Context.timed("deadcode-elimination", source.name) { Deadcode.remove(mainSymbol, core) } - - if !Context.config.optimize() then return tree; - - // (2) lift static arguments (worker/wrapper) - val lifted = Context.timed("static-argument-transformation", source.name) { - StaticArguments.transform(mainSymbol, tree) - } - - // (3) inline unique block definitions - Context.timed("inliner", source.name) { - Inline.full(Set(mainSymbol), lifted, Context.config.maxInlineSize().toInt) - } -} diff --git a/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala b/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala index c2057425c..8030e7dff 100644 --- a/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala +++ b/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala @@ -369,7 +369,7 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { (BlockType.Function(tparams, cparams, vparams, bparams, boxedResult), boxedResult, result) case _ => Context.abort("Body of a region cannot have interface type") } - val doBoxResult = coercer[Block](tBody.tpe, expectedBodyTpe) + val doBoxResult = coercer[BlockLit](tBody.tpe, expectedBodyTpe) // Create coercer for eagerly unboxing the result again val doUnboxResult = coercer(actualReturnType, expectedReturnType) val resName = TmpValue("boxedResult") diff --git a/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala b/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala index 96d11ed33..b989fb753 100644 --- a/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala +++ b/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala @@ -105,7 +105,7 @@ object PrettyPrinter extends ParenPrettyPrinter { case Select(b, field, tpe) => toDoc(b) <> "." <> toDoc(field) case Box(b, capt) => parens("box" <+> toDoc(b)) - case Run(s) => "run" <+> braces(toDoc(s)) + case Run(s) => "run" <+> block(toDoc(s)) } def argsToDoc(targs: List[core.ValueType], vargs: List[core.Pure], bargs: List[core.Block]): Doc = @@ -154,7 +154,7 @@ object PrettyPrinter extends ParenPrettyPrinter { def toDoc(d: Definition): Doc = d match { case Definition.Def(id, BlockLit(tps, cps, vps, bps, body)) => - "def" <+> toDoc(id) <> paramsToDoc(tps, vps, bps) <+> "=" <> nested(toDoc(body)) + "def" <+> toDoc(id) <> paramsToDoc(tps, vps, bps) <+> "=" <+> block(toDoc(body)) case Definition.Def(id, block) => "def" <+> toDoc(id) <+> "=" <+> toDoc(block) case Definition.Let(id, _, binding) => @@ -166,14 +166,14 @@ object PrettyPrinter extends ParenPrettyPrinter { toDoc(definitions) <> emptyline <> toDoc(rest) case Return(e) => - toDoc(e) + "return" <+> toDoc(e) case Val(Wildcard(), _, binding, body) => toDoc(binding) <> ";" <> line <> toDoc(body) case Val(id, tpe, binding, body) => - "val" <+> toDoc(id) <> ":" <+> toDoc(tpe) <+> "=" <+> toDoc(binding) <> ";" <> line <> + "val" <+> toDoc(id) <> ":" <+> toDoc(tpe) <+> "=" <+> block(toDoc(binding)) <> ";" <> line <> toDoc(body) case App(b, targs, vargs, bargs) => diff --git a/effekt/shared/src/main/scala/effekt/core/Renamer.scala b/effekt/shared/src/main/scala/effekt/core/Renamer.scala index f43b9c07b..6d90f3548 100644 --- a/effekt/shared/src/main/scala/effekt/core/Renamer.scala +++ b/effekt/shared/src/main/scala/effekt/core/Renamer.scala @@ -18,6 +18,9 @@ class Renamer(names: Names = Names(Map.empty), prefix: String = "") extends core // list of scopes that map bound symbols to their renamed variants. private var scopes: List[Map[Id, Id]] = List.empty + // Here we track ALL renamings + var renamed: Map[Id, Id] = Map.empty + private var suffix: Int = 0 def freshIdFor(id: Id): Id = @@ -28,7 +31,9 @@ class Renamer(names: Names = Names(Map.empty), prefix: String = "") extends core def withBindings[R](ids: List[Id])(f: => R): R = val before = scopes try { - scopes = ids.map { x => x -> freshIdFor(x) }.toMap :: scopes + val newScope = ids.map { x => x -> freshIdFor(x) }.toMap + scopes = newScope :: scopes + renamed = renamed ++ newScope f } finally { scopes = before } @@ -108,4 +113,8 @@ class Renamer(names: Names = Names(Map.empty), prefix: String = "") extends core object Renamer { def rename(b: Block): Block = Renamer().rewrite(b) + def rename(b: BlockLit): (BlockLit, Map[Id, Id]) = + val renamer = Renamer() + val res = renamer.rewrite(b) + (res, renamer.renamed) } diff --git a/effekt/shared/src/main/scala/effekt/core/Tree.scala b/effekt/shared/src/main/scala/effekt/core/Tree.scala index 24d4cdfe9..fef09cfee 100644 --- a/effekt/shared/src/main/scala/effekt/core/Tree.scala +++ b/effekt/shared/src/main/scala/effekt/core/Tree.scala @@ -6,6 +6,8 @@ import effekt.util.Structural import effekt.util.messages.INTERNAL_ERROR import effekt.util.messages.ErrorReporter +import scala.annotation.tailrec + /** * Tree structure of programs in our internal core representation. * @@ -304,7 +306,7 @@ enum Stmt extends Tree { case Match(scrutinee: Pure, clauses: List[(Id, BlockLit)], default: Option[Stmt]) // (Type-monomorphic?) Regions - case Region(body: Block) + case Region(body: BlockLit) case Alloc(id: Id, init: Pure, region: Id, body: Stmt) // creates a fresh state handler to model local (backtrackable) state. @@ -336,147 +338,17 @@ enum Stmt extends Tree { export Stmt.* /** - * Smart constructors to establish some normal form + * A smart constructor for `stmt.Scope` that only introduces a scope if there are bindings */ -object normal { - - def valDef(id: Id, tpe: ValueType, binding: Stmt, body: Stmt): Stmt = - (binding, body) match { - - // [[ val x = STMT; return x ]] == STMT - case (_, Stmt.Return(Pure.ValueVar(other, _))) if other == id => - binding - - // [[ val x = return EXPR; STMT ]] = [[ let x = EXPR; STMT ]] - // - // This opt is too good for JS: it blows the stack on - // recursive functions that are used to encode while... - // - // The solution to this problem is implemented in core.MakeStackSafe: - // all recursive functions that could blow the stack are trivially wrapped - // again, after optimizing. - case (Stmt.Return(expr), body) => - scope(List(Definition.Let(id, tpe, expr)), body) - - // here we are flattening scopes; be aware that this extends - // life-times of bindings! - // - // { val x = { def...; BODY }; REST } = { def ...; val x = BODY } - case (Stmt.Scope(definitions, binding), body) => - scope(definitions, valDef(id, tpe, binding, body)) - - case _ => Stmt.Val(id, tpe, binding, body) - } - - // { def f=...; { def g=...; BODY } } = { def f=...; def g; BODY } - def scope(definitions: List[Definition], body: Stmt): Stmt = body match { - case Stmt.Scope(others, body) => scope(definitions ++ others, body) - case _ => if (definitions.isEmpty) body else Stmt.Scope(definitions, body) - } - - // TODO perform record selection here, if known - def select(target: Pure, field: Id, annotatedType: ValueType): Pure = - Select(target, field, annotatedType) - - def app(callee: Block, targs: List[ValueType], vargs: List[Pure], bargs: List[Block]): Stmt = - callee match { - case b : Block.BlockLit => reduce(b, targs, vargs, bargs) - case other => Stmt.App(callee, targs, vargs, bargs) - } - - def invoke(callee: Block, method: Id, methodTpe: BlockType, targs: List[ValueType], vargs: List[Pure], bargs: List[Block]): Stmt = - callee match { - case Block.New(impl) => - val Operation(name, tps, cps, vps, bps, body) = - impl.operations.find(op => op.name == method).getOrElse { - INTERNAL_ERROR("Should not happen") - } - reduce(BlockLit(tps, cps, vps, bps, body), targs, vargs, bargs) - case other => Invoke(callee, method, methodTpe, targs, vargs, bargs) - } - - def reset(body: BlockLit): Stmt = body match { - // case BlockLit(tparams, cparams, vparams, List(prompt), - // Stmt.Shift(prompt2, body) if prompt.id == prompt2.id => ??? - case other => Stmt.Reset(body) - } - - def make(tpe: ValueType.Data, tag: Id, vargs: List[Pure]): Pure = - Pure.Make(tpe, tag, vargs) - - def pureApp(callee: Block, targs: List[ValueType], vargs: List[Pure]): Pure = - callee match { - case b : Block.BlockLit => - INTERNAL_ERROR( - """|This should not happen! - |User defined functions always have to be called with App, not PureApp. - |If this error does occur, this means this changed. - |Check `core.Transformer.makeFunctionCall` for details. - |""".stripMargin) - case other => - Pure.PureApp(callee, targs, vargs) - } - - // "match" is a keyword in Scala - def patternMatch(scrutinee: Pure, clauses: List[(Id, BlockLit)], default: Option[Stmt]): Stmt = - scrutinee match { - case Pure.Make(dataType, ctorTag, vargs) => - clauses.collectFirst { case (tag, lit) if tag == ctorTag => lit } - .map(body => app(body, Nil, vargs, Nil)) - .orElse { default }.getOrElse { sys error "Pattern not exhaustive. This should not happen" } - case other => (clauses, default) match { - // Unit-like types: there is only one case and it is just a tag. - // sc match { case Unit() => body } ==> body - case ((id, lit) :: Nil, None) if lit.vparams.isEmpty => lit.body - case _ => Match(scrutinee, clauses, default) - } - } - - - def directApp(callee: Block, targs: List[ValueType], vargs: List[Pure], bargs: List[Block]): Expr = - callee match { - case b : Block.BlockLit => run(reduce(b, targs, vargs, Nil)) - case other => DirectApp(callee, targs, vargs, bargs) - } - - def reduce(b: BlockLit, targs: List[core.ValueType], vargs: List[Pure], bargs: List[Block]): Stmt = { - - // Only bind if not already a variable!!! - var ids: Set[Id] = Set.empty - var bindings: List[Definition.Def] = Nil - var bvars: List[Block.BlockVar] = Nil - - // (1) first bind - bargs foreach { - case x: Block.BlockVar => bvars = bvars :+ x - // introduce a binding - case block => - val id = symbols.TmpBlock("blockBinding") - bindings = bindings :+ Definition.Def(id, block) - bvars = bvars :+ Block.BlockVar(id, block.tpe, block.capt) - ids += id - } - - // (2) substitute - val body = substitutions.substitute(b, targs, vargs, bvars) - - scope(bindings, body) - } - - def run(s: Stmt): Expr = s match { - case Stmt.Return(expr) => expr - case _ => Run(s) - } - - def box(b: Block, capt: Captures): Pure = b match { - case Block.Unbox(pure) => pure - case b => Box(b, capt) - } - - def unbox(p: Pure): Block = p match { - case Pure.Box(b, _) => b - case p => Unbox(p) - } +def MaybeScope(definitions: List[Definition], body: Stmt): Stmt = body match { + // flatten scopes + // { def f = ...; { def g = ...; BODY } } = { def f = ...; def g; BODY } + case Stmt.Scope(others, body) => MaybeScope(definitions ++ others, body) + + // Drop scope if empty + // { ; BODY } = BODY + case _ if definitions.isEmpty => body + case _ => Stmt.Scope(definitions, body) } /** @@ -603,6 +475,55 @@ object Tree { case (p, b) => (p, rewrite(b)) } } + + class RewriteWithContext[Ctx] extends Structural { + def id(using Ctx): PartialFunction[Id, Id] = PartialFunction.empty + def pure(using Ctx): PartialFunction[Pure, Pure] = PartialFunction.empty + def expr(using Ctx): PartialFunction[Expr, Expr] = PartialFunction.empty + def stmt(using Ctx): PartialFunction[Stmt, Stmt] = PartialFunction.empty + def defn(using Ctx): PartialFunction[Definition, Definition] = PartialFunction.empty + def block(using Ctx): PartialFunction[Block, Block] = PartialFunction.empty + def handler(using Ctx): PartialFunction[Implementation, Implementation] = PartialFunction.empty + def param(using Ctx): PartialFunction[Param, Param] = PartialFunction.empty + + def rewrite(x: Id)(using Ctx): Id = if id.isDefinedAt(x) then id(x) else x + def rewrite(p: Pure)(using Ctx): Pure = rewriteStructurally(p, pure) + def rewrite(e: Expr)(using Ctx): Expr = rewriteStructurally(e, expr) + def rewrite(s: Stmt)(using Ctx): Stmt = rewriteStructurally(s, stmt) + def rewrite(b: Block)(using Ctx): Block = rewriteStructurally(b, block) + def rewrite(d: Definition)(using Ctx): Definition = rewriteStructurally(d, defn) + def rewrite(e: Implementation)(using Ctx): Implementation = rewriteStructurally(e, handler) + def rewrite(o: Operation)(using Ctx): Operation = rewriteStructurally(o) + def rewrite(p: Param)(using Ctx): Param = rewriteStructurally(p, param) + def rewrite(p: Param.ValueParam)(using Ctx): Param.ValueParam = rewrite(p: Param).asInstanceOf[Param.ValueParam] + def rewrite(p: Param.BlockParam)(using Ctx): Param.BlockParam = rewrite(p: Param).asInstanceOf[Param.BlockParam] + def rewrite(b: ExternBody)(using Ctx): ExternBody= rewrite(b) + + def rewrite(b: BlockLit)(using Ctx): BlockLit = if block.isDefinedAt(b) then block(b).asInstanceOf else b match { + case BlockLit(tparams, cparams, vparams, bparams, body) => + BlockLit(tparams map rewrite, cparams map rewrite, vparams map rewrite, bparams map rewrite, rewrite(body)) + } + def rewrite(b: BlockVar)(using Ctx): BlockVar = if block.isDefinedAt(b) then block(b).asInstanceOf else b match { + case BlockVar(id, annotatedTpe, annotatedCapt) => BlockVar(rewrite(id), rewrite(annotatedTpe), rewrite(annotatedCapt)) + } + + def rewrite(t: ValueType)(using Ctx): ValueType = rewriteStructurally(t) + def rewrite(t: ValueType.Data)(using Ctx): ValueType.Data = rewriteStructurally(t) + + def rewrite(t: BlockType)(using Ctx): BlockType = rewriteStructurally(t) + def rewrite(t: BlockType.Interface)(using Ctx): BlockType.Interface = rewriteStructurally(t) + def rewrite(capt: Captures)(using Ctx): Captures = capt.map(rewrite) + + def rewrite(m: ModuleDecl)(using Ctx): ModuleDecl = + m match { + case ModuleDecl(path, includes, declarations, externs, definitions, exports) => + ModuleDecl(path, includes, declarations, externs, definitions.map(rewrite), exports) + } + + def rewrite(matchClause: (Id, BlockLit))(using Ctx): (Id, BlockLit) = matchClause match { + case (p, b) => (p, rewrite(b)) + } + } } enum Variable { @@ -838,6 +759,15 @@ object substitutions { case h : Hole => h } + def substitute(b: BlockLit)(using subst: Substitution): BlockLit = b match { + case BlockLit(tparams, cparams, vparams, bparams, body) => + val shadowedTypelevel = subst shadowTypes tparams shadowCaptures cparams + BlockLit(tparams, cparams, + vparams.map(p => substitute(p)(using shadowedTypelevel)), + bparams.map(p => substitute(p)(using shadowedTypelevel)), + substitute(body)(using shadowedTypelevel shadowParams (vparams ++ bparams))) + } + def substituteAsVar(id: Id)(using subst: Substitution): Id = subst.blocks.get(id) map { case BlockVar(x, _, _) => x @@ -848,19 +778,9 @@ object substitutions { block match { case BlockVar(id, tpe, capt) if subst.blocks.isDefinedAt(id) => subst.blocks(id) case BlockVar(id, tpe, capt) => BlockVar(id, substitute(tpe), substitute(capt)) - - case BlockLit(tparams, cparams, vparams, bparams, body) => - val shadowedTypelevel = subst shadowTypes tparams shadowCaptures cparams - BlockLit(tparams, cparams, - vparams.map(p => substitute(p)(using shadowedTypelevel)), - bparams.map(p => substitute(p)(using shadowedTypelevel)), - substitute(body)(using shadowedTypelevel shadowParams (vparams ++ bparams))) - - case Unbox(pure) => - Unbox(substitute(pure)) - - case New(impl) => - New(substitute(impl)) + case b: BlockLit => substitute(b) + case Unbox(pure) => Unbox(substitute(pure)) + case New(impl) => New(substitute(impl)) } def substitute(pure: Pure)(using subst: Substitution): Pure = diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/BindSubexpressions.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/BindSubexpressions.scala new file mode 100644 index 000000000..52ebfd34e --- /dev/null +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/BindSubexpressions.scala @@ -0,0 +1,199 @@ +package effekt +package core +package optimizer + +// Establishes a normal form in which every subexpression +// is explicitly named and aliasing (val x = y) is removed. +// +// let x = Cons(1, Cons(2, Cons(3, Nil()))) +// +// -> +// let x1 = Nil() +// let x2 = Cons(3, x1) +// let x3 = Cons(2, x2) +// let x = Cons(1, x3) +object BindSubexpressions { + + type Env = Map[Id, Id] + def alias(from: Id, to: Id, env: Env): Env = + env + (from -> env.getOrElse(to, to)) + + def transform(m: ModuleDecl): ModuleDecl = m match { + case ModuleDecl(path, includes, declarations, externs, definitions, exports) => + val (newDefs, env) = transformDefs(definitions)(using Map.empty) + ModuleDecl(path, includes, declarations, externs, newDefs, exports) + } + + def transformDefs(definitions: List[Definition])(using env: Env): (List[Definition], Env) = + var definitionsSoFar = List.empty[Definition] + var envSoFar = env + + definitions.foreach { + case Definition.Def(id, block) => + transform(block)(using envSoFar) match { + case Bind(Block.BlockVar(x, _, _), defs) => + definitionsSoFar ++= defs + envSoFar = alias(id, x, envSoFar) + + case Bind(other, defs) => + definitionsSoFar = definitionsSoFar ++ (defs :+ Definition.Def(id, other)) + } + case Definition.Let(id, tpe, expr) => + transform(expr)(using envSoFar) match { + case Bind(Pure.ValueVar(x, _), defs) => + definitionsSoFar ++= defs + envSoFar = alias(id, x, envSoFar) + case Bind(other, defs) => + definitionsSoFar = definitionsSoFar ++ (defs :+ Definition.Let(id, transform(tpe)(using envSoFar), other)) + } + } + (definitionsSoFar, envSoFar) + + def transform(s: Stmt)(using env: Env): Stmt = s match { + case Stmt.Scope(definitions, body) => + val (newDefs, newEnv) = transformDefs(definitions) + MaybeScope(newDefs, transform(body)(using newEnv)) + + case Stmt.App(callee, targs, vargs, bargs) => delimit { + for { + c <- transform(callee) + vs <- transformExprs(vargs) + bs <- transformBlocks(bargs) + } yield Stmt.App(c, targs.map(transform), vs, bs) + } + + case Stmt.Invoke(callee, method, methodTpe, targs, vargs, bargs) => delimit { + for { + c <- transform(callee) + vs <- transformExprs(vargs) + bs <- transformBlocks(bargs) + } yield Stmt.Invoke(c, method, transform(methodTpe), targs.map(transform), vs, bs) + } + + case Stmt.Return(expr) => transform(expr).run { res => Stmt.Return(res) } + case Stmt.Alloc(id, init, region, body) => transform(init).run { v => Stmt.Alloc(id, v, transform(region), transform(body)) } + case Stmt.Var(id, init, capture, body) => transform(init).run { v => Stmt.Var(id, v, transform(capture), transform(body)) } + case Stmt.Get(id, capt, tpe) => Stmt.Get(id, transform(capt), transform(tpe)) + case Stmt.Put(id, capt, value) => transform(value).run { v => Stmt.Put(id, transform(capt), v) } + + case Stmt.If(cond, thn, els) => transform(cond).run { c => + Stmt.If(c, transform(thn), transform(els)) + } + case Stmt.Match(scrutinee, clauses, default) => transform(scrutinee).run { sc => + Stmt.Match(sc, clauses.map { case (tag, rhs) => (tag, transform(rhs)) }, default.map(transform)) + } + + // Congruences + case Stmt.Region(body) => Stmt.Region(transform(body)) + case Stmt.Val(id, tpe, binding, body) => Stmt.Val(id, transform(tpe), transform(binding), transform(body)) + case Stmt.Reset(body) => Stmt.Reset(transform(body)) + case Stmt.Shift(prompt, body) => Stmt.Shift(transform(prompt), transform(body)) + case Stmt.Resume(k, body) => Stmt.Resume(transform(k), transform(body)) + case Stmt.Hole() => Stmt.Hole() + } + + def transform(b: Block)(using Env): Bind[Block] = b match { + case b: Block.BlockVar => pure(transform(b)) + case b: Block.BlockLit => pure(transform(b)) + case Block.New(impl) => pure(Block.New(transform(impl))) + case Block.Unbox(pure) => transform(pure) { v => bind(Block.Unbox(v)) } + } + + def transform(b: BlockLit)(using Env): BlockLit = b match { + case BlockLit(tparams, cparams, vparams, bparams, body) => + BlockLit(tparams, cparams, vparams.map(transform), bparams.map(transform), transform(body)) + } + + def transform(b: BlockVar)(using Env): BlockVar = b match { + case BlockVar(id, annotatedTpe, annotatedCapt) => + BlockVar(transform(id), transform(annotatedTpe), transform(annotatedCapt)) + } + + def transform(impl: Implementation)(using Env): Implementation = impl match { + case Implementation(interface, operations) => Implementation(transform(interface).asInstanceOf, operations.map { + case Operation(name, tparams, cparams, vparams, bparams, body) => + Operation(name, tparams, cparams, vparams.map(transform), bparams.map(transform), transform(body)) + }) + } + + def transform(p: ValueParam)(using Env): ValueParam = p match { + case ValueParam(id, tpe) => ValueParam(id, transform(tpe)) + } + def transform(p: BlockParam)(using Env): BlockParam = p match { + case BlockParam(id, tpe, capt) => BlockParam(id, transform(tpe), transform(capt)) + } + + def transform(id: Id)(using env: Env): Id = env.getOrElse(id, id) + + def transform(e: Expr)(using Env): Bind[ValueVar | Literal] = e match { + case Pure.ValueVar(id, tpe) => pure(ValueVar(transform(id), transform(tpe))) + case Pure.Literal(value, tpe) => pure(Pure.Literal(value, transform(tpe))) + + case Pure.Make(data, tag, vargs) => transformExprs(vargs) { vs => + bind(Pure.Make(data, tag, vs)) + } + case DirectApp(block, targs, vargs, bargs) => for { + b <- transform(block); + vs <- transformExprs(vargs); + bs <- transformBlocks(bargs); + res <- bind(DirectApp(b, targs.map(transform), vs, bs)) + } yield res + case Pure.PureApp(block, targs, vargs) => for { + b <- transform(block); + vs <- transformExprs(vargs); + res <- bind(Pure.PureApp(b, targs.map(transform), vs)) + } yield res + case Pure.Select(target, field, tpe) => transform(target) { v => bind(Pure.Select(v, field, transform(tpe))) } + case Pure.Box(block, capt) => transform(block) { b => bind(Pure.Box(b, transform(capt))) } + + case Run(s) => bind(Run(transform(s))) + } + + def transformExprs(es: List[Expr])(using Env): Bind[List[ValueVar | Literal]] = traverse(es)(transform) + def transformBlocks(es: List[Block])(using Env): Bind[List[Block]] = traverse(es)(transform) + + // Types + // ----- + // Types mention captures and captures might require renaming after dealiasing + def transform(tpe: ValueType)(using Env): ValueType = tpe match { + case ValueType.Var(name) => ValueType.Var(transform(name)) + case ValueType.Data(name, targs) => ValueType.Data(name, targs.map(transform)) + case ValueType.Boxed(tpe, capt) => ValueType.Boxed(transform(tpe), transform(capt)) + } + def transform(tpe: BlockType)(using Env): BlockType = tpe match { + case BlockType.Function(tparams, cparams, vparams, bparams, result) => + BlockType.Function(tparams, cparams, vparams.map(transform), bparams.map(transform), transform(result)) + case BlockType.Interface(name, targs) => + BlockType.Interface(name, targs.map(transform)) + } + def transform(captures: Captures)(using Env): Captures = captures.map(transform) + + + // Binding Monad + // ------------- + case class Bind[+A](value: A, definitions: List[Definition]) { + def run(f: A => Stmt): Stmt = MaybeScope(definitions, f(value)) + def map[B](f: A => B): Bind[B] = Bind(f(value), definitions) + def flatMap[B](f: A => Bind[B]): Bind[B] = + val Bind(result, other) = f(value) + Bind(result, definitions ++ other) + def apply[B](f: A => Bind[B]): Bind[B] = flatMap(f) + } + def pure[A](value: A): Bind[A] = Bind(value, Nil) + def bind[A](expr: Expr): Bind[ValueVar] = + val id = Id("tmp") + Bind(ValueVar(id, expr.tpe), List(Definition.Let(id, expr.tpe, expr))) + + def bind[A](block: Block): Bind[BlockVar] = + val id = Id("tmp") + Bind(BlockVar(id, block.tpe, block.capt), List(Definition.Def(id, block))) + + def delimit(b: Bind[Stmt]): Stmt = b.run(a => a) + + def traverse[S, T](l: List[S])(f: S => Bind[T]): Bind[List[T]] = + l match { + case Nil => pure(Nil) + case head :: tail => for { x <- f(head); xs <- traverse(tail)(f) } yield x :: xs + } + +} diff --git a/effekt/shared/src/main/scala/effekt/core/Deadcode.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/Deadcode.scala similarity index 68% rename from effekt/shared/src/main/scala/effekt/core/Deadcode.scala rename to effekt/shared/src/main/scala/effekt/core/optimizer/Deadcode.scala index 8765d570a..649e5c28b 100644 --- a/effekt/shared/src/main/scala/effekt/core/Deadcode.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/Deadcode.scala @@ -1,22 +1,23 @@ package effekt package core +package optimizer -import effekt.core.Block.BlockLit -import effekt.core.Pure.ValueVar -import effekt.core.normal.* - -class Deadcode(entrypoints: Set[Id], definitions: Map[Id, Definition]) extends core.Tree.Rewrite { - - val reachable = Reachable(entrypoints, definitions) +class Deadcode(reachable: Map[Id, Usage]) extends core.Tree.Rewrite { override def stmt = { // Remove local unused definitions case Scope(defs, stmt) => - scope(defs.collect { + MaybeScope(defs.collect { case d: Definition.Def if reachable.isDefinedAt(d.id) => rewrite(d) // we only keep non-pure OR reachable let bindings case d: Definition.Let if d.capt.nonEmpty || reachable.isDefinedAt(d.id) => rewrite(d) }, rewrite(stmt)) + + case Reset(body) => + rewrite(body) match { + case BlockLit(tparams, cparams, vparams, List(prompt), body) if !reachable.isDefinedAt(prompt.id) => body + case b => Stmt.Reset(b) + } } override def rewrite(m: ModuleDecl): ModuleDecl = m.copy( @@ -31,7 +32,9 @@ class Deadcode(entrypoints: Set[Id], definitions: Map[Id, Definition]) extends c object Deadcode { def remove(entrypoints: Set[Id], m: ModuleDecl): ModuleDecl = - Deadcode(entrypoints, m.definitions.map(d => d.id -> d).toMap).rewrite(m) + val reachable = Reachable(entrypoints, m) + Deadcode(reachable).rewrite(m) + def remove(entrypoint: Id, m: ModuleDecl): ModuleDecl = remove(Set(entrypoint), m) } diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/DropBindings.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/DropBindings.scala new file mode 100644 index 000000000..740586c47 --- /dev/null +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/DropBindings.scala @@ -0,0 +1,66 @@ +package effekt +package core +package optimizer + +import context.Context + +/** + * This phase drops (inlines) unique value bindings. + * + * let x = 42 + * let y = x + 1 + * let z = y * 2 + * z + * + * --> + * + * (42 + 1) * 2 + * + * This improves the performance for JS on some benchmarks (mostly match_options, sum_range, and parsing_dollars). + */ +object DropBindings extends Phase[CoreTransformed, CoreTransformed] { + + val phaseName: String = "drop-bindings" + + def run(input: CoreTransformed)(using C: Context): Option[CoreTransformed] = + input match { + case CoreTransformed(source, tree, mod, core) => + val main = C.checkMain(mod) + Some(CoreTransformed(source, tree, mod, apply(Set(main), core))) + } + + def apply(entrypoints: Set[Id], m: ModuleDecl): ModuleDecl = + dropping.rewrite(m)(using DropContext(Reachable(entrypoints, m), Map.empty)) + + private case class DropContext( + usage: Map[Id, Usage], + definitions: Map[Id, Pure] + ) { + def updated(id: Id, p: Pure): DropContext = this.copy(definitions = definitions.updated(id, p)) + } + + private def hasDefinition(id: Id)(using C: DropContext) = C.definitions.isDefinedAt(id) + private def definitionOf(id: Id)(using C: DropContext): Pure = C.definitions(id) + private def usedOnce(id: Id)(using C: DropContext) = C.usage.get(id).contains(Usage.Once) + private def currentContext(using C: DropContext): C.type = C + + private object dropping extends Tree.RewriteWithContext[DropContext] { + + override def pure(using DropContext) = { + case Pure.ValueVar(id, tpe) if usedOnce(id) && hasDefinition(id) => definitionOf(id) + } + + override def stmt(using DropContext) = { + case Stmt.Scope(defs, body) => + var contextSoFar = currentContext + val ds = defs.flatMap { + case Definition.Let(id, tpe, p: Pure) if usedOnce(id) => + val transformed = rewrite(p)(using contextSoFar) + contextSoFar = contextSoFar.updated(id, transformed) + None + case d => Some(rewrite(d)(using contextSoFar)) + } + MaybeScope(ds, rewrite(body)(using contextSoFar)) + } + } +} diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala new file mode 100644 index 000000000..2744f255b --- /dev/null +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala @@ -0,0 +1,402 @@ +package effekt +package core +package optimizer + +import effekt.util.messages.INTERNAL_ERROR + +import scala.annotation.tailrec +import scala.collection.mutable + +/** + * Removes "cuts", that is it performs a step of computation if enough information + * is available. + * + * def foo(n: Int) = return n + 1 + * + * foo(42) + * + * becomes + * + * def foo(n: Int) = return n + 1 + * return 42 + 1 + * + * removing the overhead of the function call. Under the following conditions, + * cuts are _not_ removed: + * + * - the definition is recursive + * - inlining would exceed the maxInlineSize + * + * If the function is called _exactly once_, it is inlined regardless of the maxInlineSize. + */ +object Normalizer { normal => + + case class Context( + blocks: Map[Id, Block], + exprs: Map[Id, Expr], + decls: DeclarationContext, // for field selection + usage: mutable.Map[Id, Usage], // mutable in order to add new information after renaming + maxInlineSize: Int // to control inlining and avoid code bloat + ) { + def bind(id: Id, expr: Expr): Context = copy(exprs = exprs + (id -> expr)) + def bind(id: Id, block: Block): Context = copy(blocks = blocks + (id -> block)) + } + + private def blockFor(id: Id)(using ctx: Context): Option[Block] = + ctx.blocks.get(id) + + private def exprFor(id: Id)(using ctx: Context): Option[Expr] = + ctx.exprs.get(id) + + private def isRecursive(id: Id)(using ctx: Context): Boolean = + ctx.usage.get(id) match { + case Some(value) => value == Usage.Recursive + // We assume it is recursive, if (for some reason) we do not have information; + // since reducing might diverge, otherwise. + // + // This is, however, a strange case since this means we call a function we deemed unreachable. + // It _can_ happen, for instance, by updating the usage (subtracting) and not deadcode eliminating. + // This is the case for examples/pos/bidirectional/scheduler.effekt + case None => true // sys error s"No info for ${id}" + } + + private def isOnce(id: Id)(using ctx: Context): Boolean = + ctx.usage.get(id) match { + case Some(value) => value == Usage.Once + case None => false + } + + def normalize(entrypoints: Set[Id], m: ModuleDecl, maxInlineSize: Int): ModuleDecl = { + // usage information is used to detect recursive functions (and not inline them) + val usage = Reachable(entrypoints, m) + + val defs = m.definitions.collect { + case Definition.Def(id, block) => id -> block + }.toMap + val context = Context(defs, Map.empty, DeclarationContext(m.declarations, m.externs), mutable.Map.from(usage), maxInlineSize) + + val (normalizedDefs, _) = normalize(m.definitions)(using context) + m.copy(definitions = normalizedDefs) + } + + def normalize(definitions: List[Definition])(using ctx: Context): (List[Definition], Context) = + var contextSoFar = ctx + val defs = definitions.map { + case Definition.Def(id, block) => + val normalized = active(block)(using contextSoFar).dealiased + contextSoFar = contextSoFar.bind(id, normalized) + Definition.Def(id, normalized) + case Definition.Let(id, tpe, expr) => + val normalized = active(expr)(using contextSoFar) + contextSoFar = contextSoFar.bind(id, normalized) + Definition.Let(id, tpe, normalized) + } + (defs, contextSoFar) + + private enum NormalizedBlock { + case Known(b: BlockLit | New | Unbox, boundBy: Option[BlockVar]) + case Unknown(b: BlockVar) + + def dealiased: Block = this match { + case NormalizedBlock.Known(b, boundBy) => b + case NormalizedBlock.Unknown(b) => b + } + def shared: Block = this match { + case NormalizedBlock.Known(b, boundBy) => boundBy.getOrElse(b) + case NormalizedBlock.Unknown(b) => b + } + } + + /** + * This is a bit tricky: depending on the call-site of `active` + * we either want to find a redex (BlockLit | New), maximally dealias (in def bindings), + * discover the outmost Unbox (when boxing again), or preserve some sharing otherwise. + * + * A good testcase to look at for this is: + * examples/pos/capture/regions.effekt + */ + private def active[R](b: Block)(using C: Context): NormalizedBlock = + normalize(b) match { + case b: Block.BlockLit => NormalizedBlock.Known(b, None) + case b @ Block.New(impl) => NormalizedBlock.Known(b, None) + + case x @ Block.BlockVar(id, annotatedTpe, annotatedCapt) => blockFor(id) match { + case Some(b: (BlockLit | New | Unbox)) => NormalizedBlock.Known(b, Some(x)) + case _ => NormalizedBlock.Unknown(x) + } + case Block.Unbox(pure) => active(pure) match { + case Pure.Box(b, annotatedCapture) => active(b) + case other => NormalizedBlock.Known(Block.Unbox(pure), None) + } + } + + // TODO for `New` we should track how often each operation is used, not the object itself + // to decide inlining. + private def shouldInline(b: BlockLit, boundBy: Option[BlockVar])(using C: Context): Boolean = boundBy match { + case Some(id) if isRecursive(id.id) => false + case Some(id) => isOnce(id.id) || b.body.size <= C.maxInlineSize + case None => true + } + + private def active(e: Expr)(using Context): Expr = + normalize(e) match { + case x @ Pure.ValueVar(id, annotatedType) => exprFor(id) match { + case Some(p: Pure.Make) => p + case Some(p: Pure.Literal) => p + case Some(p: Pure.Box) => p + // We only inline non side-effecting expressions + case Some(other) if other.capt.isEmpty => other + case _ => x // stuck + } + case other => other // stuck + } + + def normalize(d: Definition)(using C: Context): Definition = d match { + case Definition.Def(id, block) => Definition.Def(id, normalize(block)) + case Definition.Let(id, tpe, binding) => Definition.Let(id, tpe, normalize(binding)) + } + + def normalize(s: Stmt)(using C: Context): Stmt = s match { + + case Stmt.Scope(definitions, body) => + val (defs, ctx) = normalize(definitions) + normal.Scope(defs, normalize(body)(using ctx)) + + // Redexes + // ------- + case Stmt.App(b, targs, vargs, bargs) => + active(b) match { + case NormalizedBlock.Known(b: BlockLit, boundBy) if shouldInline(b, boundBy) => + reduce(b, targs, vargs.map(normalize), bargs.map(normalize)) + case normalized => + Stmt.App(normalized.shared, targs, vargs.map(normalize), bargs.map(normalize)) + } + + case Stmt.Invoke(b, method, methodTpe, targs, vargs, bargs) => + active(b) match { + case n @ NormalizedBlock.Known(Block.New(impl), boundBy) => + selectOperation(impl, method) match { + case b: BlockLit if shouldInline(b, boundBy) => reduce(b, targs, vargs.map(normalize), bargs.map(normalize)) + case _ => Stmt.Invoke(n.shared, method, methodTpe, targs, vargs.map(normalize), bargs.map(normalize)) + } + + case normalized => + Stmt.Invoke(normalized.shared, method, methodTpe, targs, vargs.map(normalize), bargs.map(normalize)) + } + + case Stmt.Match(scrutinee, clauses, default) => active(scrutinee) match { + case Pure.Make(data, tag, vargs) if clauses.exists { case (id, _) => id == tag } => + val clause: BlockLit = clauses.collectFirst { case (id, cl) if id == tag => cl }.get + normalize(reduce(clause, Nil, vargs.map(normalize), Nil)) + case Pure.Make(data, tag, vargs) if default.isDefined => + normalize(default.get) + case _ => + val normalized = normalize(scrutinee) + Stmt.Match(normalized, clauses.map { case (id, value) => id -> normalize(value) }, default.map(normalize)) + } + + // [[ if (true) stmt1 else stmt2 ]] = [[ stmt1 ]] + case Stmt.If(cond, thn, els) => active(cond) match { + case Pure.Literal(true, annotatedType) => normalize(thn) + case Pure.Literal(false, annotatedType) => normalize(els) + case _ => If(normalize(cond), normalize(thn), normalize(els)) + } + + case Stmt.Val(id, tpe, binding, body) => + + def normalizeVal(id: Id, tpe: ValueType, binding: Stmt, body: Stmt): Stmt = binding match { + + // [[ val x = return e; s ]] = let x = [[ e ]]; [[ s ]] + case Stmt.Return(expr) => + normal.Scope(List(Definition.Let(id, tpe, expr)), + normalize(body)(using C.bind(id, expr))) + + // Commute val and bindings + // [[ val x = { def f = ...; let y = ...; STMT }; STMT ]] = def f = ...; let y = ...; val x = STMT; STMT + case Stmt.Scope(ds, bodyBinding) => + normal.Scope(ds, normalizeVal(id, tpe, bodyBinding, body)) + + // Flatten vals. This should be non-leaking since we use garbage free refcounting. + // [[ val x = { val y = stmt1; stmt2 }; stmt3 ]] = [[ val y = stmt1; val x = stmt2; stmt3 ]] + case Stmt.Val(id2, tpe2, binding2, body2) => + normalizeVal(id2, tpe2, binding2, Stmt.Val(id, tpe, body2, body)) + + + // [[ val x = { var y in r = e; stmt1 }; stmt2 ]] = var y in r = e; [[ val x = stmt1; stmt2 ]] + case Stmt.Alloc(id2, init2, region2, body2) => + Stmt.Alloc(id2, init2, region2, normalizeVal(id, tpe, body2, body)) + + // [[ val x = stmt; return x ]] = [[ stmt ]] + case other => normalize(body) match { + case Stmt.Return(x: ValueVar) if x.id == id => other + case normalizedBody => Stmt.Val(id, tpe, other, normalizedBody) + } + } + normalizeVal(id, tpe, normalize(binding), body) + + + // "Congruences" + // ------------- + + case Stmt.Reset(body) => Stmt.Reset(normalize(body)) + case Stmt.Shift(prompt, body) => Shift(prompt, normalize(body)) + case Stmt.Return(expr) => Return(normalize(expr)) + case Stmt.Alloc(id, init, region, body) => Alloc(id, normalize(init), region, normalize(body)) + case Stmt.Resume(k, body) => Resume(k, normalize(body)) + case Stmt.Region(body) => Region(normalize(body)) + case Stmt.Var(id, init, capture, body) => Stmt.Var(id, normalize(init), capture, normalize(body)) + case Stmt.Get(id, capt, tpe) => Stmt.Get(id, capt, tpe) + case Stmt.Put(id, capt, value) => Stmt.Put(id, capt, normalize(value)) + case Stmt.Hole() => s + } + def normalize(b: BlockLit)(using Context): BlockLit = + b match { + case BlockLit(tparams, cparams, vparams, bparams, body) => + BlockLit(tparams, cparams, vparams, bparams, normalize(body)) + } + + def normalize(b: Block)(using Context): Block = b match { + case b @ Block.BlockVar(id, _, _) => b + case b @ Block.BlockLit(tparams, cparams, vparams, bparams, body) => normalize(b) + + // [[ unbox (box b) ]] = [[ b ]] + case Block.Unbox(pure) => normal.Unbox(normalize(pure)) + case Block.New(impl) => New(normalize(impl)) + } + + def normalize(s: Implementation)(using Context): Implementation = + s match { + case Implementation(interface, operations) => Implementation(interface, operations.map { op => + op.copy(body = normalize(op.body)) + }) + } + + def normalize(p: Pure)(using ctx: Context): Pure = p match { + // [[ Constructor(f = v).f ]] = [[ v ]] + case Pure.Select(target, field, annotatedType) => active(target) match { + case Pure.Make(datatype, tag, fields) => + val constructor = ctx.decls.findConstructor(tag).get + val expr = (constructor.fields zip fields).collectFirst { case (f, expr) if f.id == field => expr }.get + normalize(expr) + case _ => Pure.Select(normalize(target), field, annotatedType) + } + + // [[ box (unbox e) ]] = [[ e ]] + case Pure.Box(b, annotatedCapture) => active(b) match { + case NormalizedBlock.Known(Unbox(p), boundBy) => p + case _ => normal.Box(normalize(b), annotatedCapture) + } + + // congruences + case Pure.PureApp(b, targs, vargs) => Pure.PureApp(normalize(b), targs, vargs.map(normalize)) + case Pure.Make(data, tag, vargs) => Pure.Make(data, tag, vargs.map(normalize)) + case Pure.ValueVar(id, annotatedType) => p + case Pure.Literal(value, annotatedType) => p + } + + def normalize(e: Expr)(using Context): Expr = e match { + case DirectApp(b, targs, vargs, bargs) => DirectApp(normalize(b), targs, vargs.map(normalize), bargs.map(normalize)) + + // [[ run (return e) ]] = [[ e ]] + case Run(s) => normal.Run(normalize(s)) + + case pure: Pure => normalize(pure) + } + + + // Smart Constructors + // ------------------ + @tailrec + private def Scope(definitions: List[Definition], body: Stmt): Stmt = body match { + + // flatten scopes + // { def f = ...; { def g = ...; BODY } } = { def f = ...; def g; BODY } + case Stmt.Scope(others, body) => normal.Scope(definitions ++ others, body) + + // commute bindings + // let x = run { let y = e; s } = let y = e; let x = run { s } + case _ => if (definitions.isEmpty) body else { + var defsSoFar: List[Definition] = Nil + + definitions.foreach { + case Definition.Let(id, tpe, Run(Stmt.Scope(ds, body))) => + defsSoFar = defsSoFar ++ (ds :+ Definition.Let(id, tpe, normal.Run(body))) + case d => defsSoFar = defsSoFar :+ d + } + Stmt.Scope(defsSoFar, body) + } + } + + private def Run(s: Stmt): Expr = s match { + + // run { let x = e; return x } = e + case Stmt.Scope(Definition.Let(id1, _, binding) :: Nil, Stmt.Return(Pure.ValueVar(id2, _))) if id1 == id2 => + binding + + // run { return e } = e + case Stmt.Return(expr) => expr + + case _ => core.Run(s) + } + + // box (unbox p) = p + private def Box(b: Block, capt: Captures): Pure = b match { + case Block.Unbox(pure) => pure + case b => Pure.Box(b, capt) + } + + // unbox (box b) = b + private def Unbox(p: Pure): Block = p match { + case Pure.Box(b, _) => b + case p => Block.Unbox(p) + } + + + // Helpers for beta-reduction + // -------------------------- + + private def reduce(b: BlockLit, targs: List[core.ValueType], vargs: List[Pure], bargs: List[Block])(using C: Context): Stmt = { + // To update usage information + val usage = C.usage + def copyUsage(from: Id, to: Id) = usage.get(from) match { + case Some(info) => usage.update(to, info) + case None => () + } + + // Only bind if not already a variable!!! + var ids: Set[Id] = Set.empty + var bindings: List[Definition.Def] = Nil + var bvars: List[Block.BlockVar] = Nil + + // (1) first bind + (b.bparams zip bargs) foreach { + case (bparam, x: Block.BlockVar) => + + // Update usage: u1 + (u2 - 1) + usage.update(x.id, usage.getOrElse(bparam.id, Usage.Never) + usage.getOrElse(x.id, Usage.Never).decrement) + bvars = bvars :+ x + // introduce a binding + case (bparam, block) => + val id = symbols.TmpBlock("blockBinding") + bindings = bindings :+ Definition.Def(id, block) + bvars = bvars :+ Block.BlockVar(id, block.tpe, block.capt) + copyUsage(bparam.id, id) + ids += id + } + + val (renamedLit: BlockLit, renamedIds) = Renamer.rename(b) + + renamedIds.foreach(copyUsage) + + val newUsage = usage.collect { case (id, usage) if util.show(id) contains "foreach" => (id, usage) } + + // (2) substitute + val body = substitutions.substitute(renamedLit, targs, vargs, bvars) + + normalize(normal.Scope(bindings, body)) + } + + private def selectOperation(impl: Implementation, method: Id): Block.BlockLit = + impl.operations.collectFirst { + case Operation(name, tps, cps, vps, bps, body) if name == method => BlockLit(tps, cps, vps, bps, body): Block.BlockLit + }.getOrElse { INTERNAL_ERROR("Should not happen") } +} diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/Optimizer.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/Optimizer.scala new file mode 100644 index 000000000..8d0d6d029 --- /dev/null +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/Optimizer.scala @@ -0,0 +1,54 @@ +package effekt +package core +package optimizer + +import effekt.PhaseResult.CoreTransformed +import effekt.context.Context + +import kiama.util.Source + +object Optimizer extends Phase[CoreTransformed, CoreTransformed] { + + val phaseName: String = "core-optimizer" + + def run(input: CoreTransformed)(using Context): Option[CoreTransformed] = + input match { + case CoreTransformed(source, tree, mod, core) => + val term = Context.checkMain(mod) + val optimized = Context.timed("optimize", source.name) { optimize(source, term, core) } + Some(CoreTransformed(source, tree, mod, optimized)) + } + + def optimize(source: Source, mainSymbol: symbols.Symbol, core: ModuleDecl)(using Context): ModuleDecl = + + var tree = core + + // (1) first thing we do is simply remove unused definitions (this speeds up all following analysis and rewrites) + tree = Context.timed("deadcode-elimination", source.name) { + Deadcode.remove(mainSymbol, tree) + } + + if !Context.config.optimize() then return tree; + + // (2) lift static arguments + tree = Context.timed("static-argument-transformation", source.name) { + StaticArguments.transform(mainSymbol, tree) + } + + def normalize(m: ModuleDecl) = { + val anfed = BindSubexpressions.transform(m) + val normalized = Normalizer.normalize(Set(mainSymbol), anfed, Context.config.maxInlineSize().toInt) + Deadcode.remove(mainSymbol, normalized) + } + + // (3) normalize once and remove beta redexes + tree = Context.timed("normalize-1", source.name) { normalize(tree) } + + // (4) optimize continuation capture in the tail-resumptive case + tree = Context.timed("tail-resumptions", source.name) { RemoveTailResumptions(tree) } + + // (5) normalize again to clean up and remove new redexes + tree = Context.timed("normalize-2", source.name) { normalize(tree) } + + tree +} diff --git a/effekt/shared/src/main/scala/effekt/core/Reachable.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala similarity index 70% rename from effekt/shared/src/main/scala/effekt/core/Reachable.scala rename to effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala index 3d51c94f0..f0eb439ed 100644 --- a/effekt/shared/src/main/scala/effekt/core/Reachable.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala @@ -1,5 +1,6 @@ package effekt package core +package optimizer /** * A simple reachability analysis. @@ -10,35 +11,35 @@ class Reachable( var seen: Set[Id] ) { + private def update(id: Id, u: Usage): Unit = reachable = reachable.updated(id, u) + private def usage(id: Id): Usage = reachable.getOrElse(id, Usage.Never) + def process(d: Definition)(using defs: Map[Id, Definition]): Unit = - if stack.contains(d.id) then - reachable = reachable.updated(d.id, Usage.Recursive) + if stack.contains(d.id) then update(d.id, Usage.Recursive) else d match { case Definition.Def(id, block) => seen = seen + id + val before = stack stack = id :: stack + process(block) stack = before case Definition.Let(id, _, binding) => seen = seen + id + process(binding) } def process(id: Id)(using defs: Map[Id, Definition]): Unit = if (stack.contains(id)) { - reachable = reachable.updated(id, Usage.Recursive) + update(id, Usage.Recursive) return; } - val count = reachable.get(id) match { - case Some(Usage.Once) => Usage.Many - case Some(Usage.Many) => Usage.Many - case Some(Usage.Recursive) => Usage.Recursive - case None => Usage.Once - } - reachable = reachable.updated(id, count) + update(id, usage(id) + Usage.Once) + if (!seen.contains(id)) { defs.get(id).foreach(process) } @@ -57,9 +58,13 @@ class Reachable( definitions.foreach { case d: Definition.Def => currentDefs += d.id -> d // recursive - process(d)(using currentDefs) + // Do NOT process them here, since this would mean the definition is used + // process(d)(using currentDefs) case d: Definition.Let => - process(d)(using currentDefs) + // DO only process if NOT pure + if (d.binding.capt.nonEmpty) { + process(d)(using currentDefs) + } currentDefs += d.id -> d // non-recursive } process(body)(using currentDefs) @@ -111,32 +116,42 @@ class Reachable( def process(i: Implementation)(using defs: Map[Id, Definition]): Unit = i.operations.foreach { op => process(op.body) } - } object Reachable { - def apply(entrypoints: Set[Id], definitions: Map[Id, Definition]): Map[Id, Usage] = { - val analysis = new Reachable(Map.empty, Nil, Set.empty) - entrypoints.foreach(d => analysis.process(d)(using definitions)) - analysis.reachable - } + def apply(entrypoints: Set[Id], m: ModuleDecl): Map[Id, Usage] = { + val definitions = m.definitions.map(d => d.id -> d).toMap + val initialUsage = entrypoints.map { id => id -> Usage.Recursive }.toMap + val analysis = new Reachable(initialUsage, Nil, Set.empty) - def apply(m: ModuleDecl): Map[Id, Usage] = { - val analysis = new Reachable(Map.empty, Nil, Set.empty) - val defs = m.definitions.map(d => d.id -> d).toMap - m.definitions.foreach(d => analysis.process(d)(using defs)) - analysis.reachable - } + entrypoints.foreach(d => analysis.process(d)(using definitions)) - def apply(s: Stmt.Scope): Map[Id, Usage] = { - val analysis = new Reachable(Map.empty, Nil, Set.empty) - analysis.process(s)(using Map.empty) analysis.reachable } } enum Usage { + case Never case Once case Many case Recursive + + def +(other: Usage): Usage = (this, other) match { + case (Usage.Never, other) => other + case (other, Usage.Never) => other + case (other, Usage.Recursive) => Usage.Recursive + case (Usage.Recursive, other) => Usage.Recursive + case (Usage.Once, Usage.Once) => Usage.Many + case (Usage.Many, Usage.Many) => Usage.Many + case (Usage.Many, Usage.Once) => Usage.Many + case (Usage.Once, Usage.Many) => Usage.Many + } + + // -1 + def decrement: Usage = this match { + case Usage.Never => Usage.Never + case Usage.Once => Usage.Never + case Usage.Many => Usage.Many + case Usage.Recursive => Usage.Recursive + } } diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/RemoveTailResumptions.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/RemoveTailResumptions.scala new file mode 100644 index 000000000..bf6544a70 --- /dev/null +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/RemoveTailResumptions.scala @@ -0,0 +1,73 @@ +package effekt +package core +package optimizer + +object RemoveTailResumptions { + + def apply(m: ModuleDecl): ModuleDecl = removal.rewrite(m) + + object removal extends Tree.Rewrite { + override def stmt: PartialFunction[Stmt, Stmt] = { + case Stmt.Shift(prompt, BlockLit(tparams, cparams, vparams, List(k), body)) if tailResumptive(k.id, body) => + removeTailResumption(k.id, body) + case Stmt.Shift(prompt, body) => Shift(prompt, rewrite(body)) + } + } + + // A simple syntactic check whether this stmt is tailresumptive in k + def tailResumptive(k: Id, stmt: Stmt): Boolean = + def freeInStmt(stmt: Stmt): Boolean = Variables.free(stmt).containsBlock(k) + def freeInExpr(expr: Expr): Boolean = Variables.free(expr).containsBlock(k) + def freeInDef(definition: Definition): Boolean = Variables.free(definition).containsBlock(k) + + stmt match { + case Stmt.Scope(definitions, body) => definitions.forall { d => !freeInDef(d) } && tailResumptive(k, body) + case Stmt.Return(expr) => false + case Stmt.Val(id, annotatedTpe, binding, body) => tailResumptive(k, body) && !freeInStmt(binding) + case Stmt.App(callee, targs, vargs, bargs) => false + case Stmt.Invoke(callee, method, methodTpe, targs, vargs, bargs) => false + case Stmt.If(cond, thn, els) => !freeInExpr(cond) && tailResumptive(k, thn) && tailResumptive(k, els) + // Interestingly, we introduce a join point making this more difficult to implement properly + case Stmt.Match(scrutinee, clauses, default) => !freeInExpr(scrutinee) && clauses.forall { + case (_, BlockLit(tparams, cparams, vparams, bparams, body)) => tailResumptive(k, body) + } && default.forall { stmt => tailResumptive(k, stmt) } + case Stmt.Region(BlockLit(tparams, cparams, vparams, bparams, body)) => tailResumptive(k, body) + case Stmt.Alloc(id, init, region, body) => tailResumptive(k, body) && !freeInExpr(init) + case Stmt.Var(id, init, capture, body) => tailResumptive(k, body) && !freeInExpr(init) + case Stmt.Get(id, annotatedCapt, annotatedTpe) => false + case Stmt.Put(id, annotatedCapt, value) => false + case Stmt.Reset(BlockLit(tparams, cparams, vparams, bparams, body)) => tailResumptive(k, body) // is this correct? + case Stmt.Shift(prompt, body) => false + case Stmt.Resume(k2, body) => k2.id == k // what if k is free in body? + case Stmt.Hole() => true + } + + def removeTailResumption(k: Id, stmt: Stmt): Stmt = stmt match { + case Stmt.Scope(definitions, body) => Stmt.Scope(definitions, removeTailResumption(k, body)) + case Stmt.Val(id, annotatedTpe, binding, body) => Stmt.Val(id, annotatedTpe, binding, removeTailResumption(k, body)) + case Stmt.If(cond, thn, els) => Stmt.If(cond, removeTailResumption(k, thn), removeTailResumption(k, els)) + case Stmt.Match(scrutinee, clauses, default) => Stmt.Match(scrutinee, clauses.map { + case (tag, block) => tag -> removeTailResumption(k, block) + }, default.map(removeTailResumption(k, _))) + case Stmt.Region(body : BlockLit) => + Stmt.Region(removeTailResumption(k, body)) + case Stmt.Alloc(id, init, region, body) => Stmt.Alloc(id, init, region, removeTailResumption(k, body)) + case Stmt.Var(id, init, capture, body) => Stmt.Var(id, init, capture, removeTailResumption(k, body)) + case Stmt.Reset(body) => Stmt.Reset(removeTailResumption(k, body)) + case Stmt.Resume(k2, body) if k2.id == k => body + + case Stmt.Resume(k, body) => stmt + case Stmt.Shift(prompt, body) => stmt + case Stmt.Hole() => stmt + case Stmt.Return(expr) => stmt + case Stmt.App(callee, targs, vargs, bargs) => stmt + case Stmt.Invoke(callee, method, methodTpe, targs, vargs, bargs) => stmt + case Stmt.Get(id, annotatedCapt, annotatedTpe) => stmt + case Stmt.Put(id, annotatedCapt, value) => stmt + } + + def removeTailResumption(k: Id, block: BlockLit): BlockLit = block match { + case BlockLit(tparams, cparams, vparams, bparams, body) => + BlockLit(tparams, cparams, vparams, bparams, removeTailResumption(k, body)) + } +} diff --git a/effekt/shared/src/main/scala/effekt/core/StaticArguments.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/StaticArguments.scala similarity index 80% rename from effekt/shared/src/main/scala/effekt/core/StaticArguments.scala rename to effekt/shared/src/main/scala/effekt/core/optimizer/StaticArguments.scala index 4aabbf42d..3a8e1214b 100644 --- a/effekt/shared/src/main/scala/effekt/core/StaticArguments.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/StaticArguments.scala @@ -1,8 +1,9 @@ package effekt package core +package optimizer -import effekt.core.normal.* import scala.collection.mutable + import effekt.core.Type.returnType /** @@ -34,9 +35,12 @@ object StaticArguments { types.forall(x => x) && (values.exists(x => x) || blocks.exists(x => x)) } - def dropStatic[A](isStatic: List[Boolean], arguments: List[A]): List[A] = + private def dropStatic[A](isStatic: List[Boolean], arguments: List[A]): List[A] = (isStatic zip arguments).collect { case (false, arg) => arg } + private def selectStatic[A](isStatic: List[Boolean], arguments: List[A]): List[A] = + (isStatic zip arguments).collect { case (true, arg) => arg } + /** * Wraps the definition in another function, abstracting arguments along the way. * For example: @@ -58,7 +62,7 @@ object StaticArguments { def wrapDefinition(id: Id, blockLit: BlockLit)(using ctx: StaticArgumentsContext): Definition.Def = val IsStatic(staticT, staticV, staticB) = ctx.statics(id) - assert(staticT.forall(x => x), "Can only apply the worker-wrapper translation, if all type arguments are static.") + assert(staticT.forall(x => x), "Can only apply the static arguments translation, if all type arguments are static.") val workerType = BlockType.Function( dropStatic(staticT, blockLit.tparams), // should always be empty! @@ -69,11 +73,8 @@ object StaticArguments { blockLit.returnType ) - val workerVar: Block.BlockVar = BlockVar(Id(id.name.name + "_worker"), workerType, blockLit.capt) - ctx.workers(id) = workerVar - // fresh params for the wrapper function and its invocation - // note: only freshen params if not static to prevent duplicates + // note: only freshen non-static params to prevent duplicates val freshCparams: List[Id] = (staticB zip blockLit.cparams).map { case (true, param) => param case (false, param) => Id(param) @@ -82,11 +83,17 @@ object StaticArguments { case (true, param) => param case (false, ValueParam(id, tpe)) => ValueParam(Id(id), tpe) } - val freshBparams: List[BlockParam] = (staticB zip blockLit.bparams).map { - case (true, param) => param - case (false, BlockParam(id, tpe, capt)) => BlockParam(Id(id), tpe, capt) + val freshBparams: List[BlockParam] = (staticB zip blockLit.bparams zip freshCparams).map { + case ((true, param), capt) => param + case ((false, BlockParam(id, tpe, _)), capt) => BlockParam(Id(id), tpe, Set(capt)) } + // the worker now closes over the static block arguments (`c` in the example above): + val newCapture = blockLit.capt ++ selectStatic(staticB, freshCparams).toSet + + val workerVar: Block.BlockVar = BlockVar(Id(id.name.name + "_worker"), workerType, newCapture) + ctx.workers(id) = workerVar + Definition.Def(id, BlockLit( blockLit.tparams, freshCparams, @@ -120,22 +127,22 @@ object StaticArguments { def rewrite(s: Stmt)(using C: StaticArgumentsContext): Stmt = s match { case Stmt.Scope(definitions, body) => - scope(rewrite(definitions), rewrite(body)) + MaybeScope(rewrite(definitions), rewrite(body)) case Stmt.App(b, targs, vargs, bargs) => b match { // if arguments are static && recursive call: call worker with reduced arguments case BlockVar(id, annotatedTpe, annotatedCapt) if hasStatics(id) && within(id) => val IsStatic(staticT, staticV, staticB) = C.statics(id) - app(C.workers(id), + Stmt.App(C.workers(id), dropStatic(staticT, targs), dropStatic(staticV, vargs).map(rewrite), dropStatic(staticB, bargs).map(rewrite)) - case _ => app(rewrite(b), targs, vargs.map(rewrite), bargs.map(rewrite)) + case _ => Stmt.App(rewrite(b), targs, vargs.map(rewrite), bargs.map(rewrite)) } case Stmt.Invoke(b, method, methodTpe, targs, vargs, bargs) => - invoke(rewrite(b), method, methodTpe, targs, vargs.map(rewrite), bargs.map(rewrite)) + Stmt.Invoke(rewrite(b), method, methodTpe, targs, vargs.map(rewrite), bargs.map(rewrite)) case Stmt.Reset(body) => rewrite(body) match { @@ -144,10 +151,9 @@ object StaticArguments { // congruences case Stmt.Return(expr) => Return(rewrite(expr)) - case Stmt.Val(id, tpe, binding, body) => valDef(id, tpe, rewrite(binding), rewrite(body)) + case Stmt.Val(id, tpe, binding, body) => Stmt.Val(id, tpe, rewrite(binding), rewrite(body)) case Stmt.If(cond, thn, els) => If(rewrite(cond), rewrite(thn), rewrite(els)) - case Stmt.Match(scrutinee, clauses, default) => - patternMatch(rewrite(scrutinee), clauses.map { case (id, value) => id -> rewrite(value) }, default.map(rewrite)) + case Stmt.Match(scrutinee, clauses, default) => Stmt.Match(rewrite(scrutinee), clauses.map { case (id, value) => id -> rewrite(value) }, default.map(rewrite)) case Stmt.Alloc(id, init, region, body) => Alloc(id, rewrite(init), region, rewrite(body)) case Stmt.Shift(prompt, body) => Shift(prompt, rewrite(body)) case Stmt.Resume(k, body) => Resume(k, rewrite(body)) @@ -155,7 +161,7 @@ object StaticArguments { case Stmt.Var(id, init, capture, body) => Stmt.Var(id, rewrite(init), capture, rewrite(body)) case Stmt.Get(id, capt, tpe) => Stmt.Get(id, capt, tpe) case Stmt.Put(id, capt, value) => Stmt.Put(id, capt, rewrite(value)) - case Stmt.Hole() => s + case Stmt.Hole() => Stmt.Hole() } def rewrite(b: BlockLit)(using StaticArgumentsContext): BlockLit = b match { @@ -168,8 +174,8 @@ object StaticArguments { // congruences case b @ Block.BlockLit(tparams, cparams, vparams, bparams, body) => rewrite(b) - case Block.Unbox(pure) => unbox(rewrite(pure)) - case Block.New(impl) => New(rewrite(impl)) + case Block.Unbox(pure) => Block.Unbox(rewrite(pure)) + case Block.New(impl) => Block.New(rewrite(impl)) } def rewrite(s: Implementation)(using StaticArgumentsContext): Implementation = @@ -180,21 +186,21 @@ object StaticArguments { } def rewrite(p: Pure)(using StaticArgumentsContext): Pure = p match { - case Pure.PureApp(b, targs, vargs) => pureApp(rewrite(b), targs, vargs.map(rewrite)) - case Pure.Make(data, tag, vargs) => make(data, tag, vargs.map(rewrite)) + case Pure.PureApp(b, targs, vargs) => Pure.PureApp(rewrite(b), targs, vargs.map(rewrite)) + case Pure.Make(data, tag, vargs) => Pure.Make(data, tag, vargs.map(rewrite)) case x @ Pure.ValueVar(id, annotatedType) => x // congruences case Pure.Literal(value, annotatedType) => p - case Pure.Select(target, field, annotatedType) => select(rewrite(target), field, annotatedType) - case Pure.Box(b, annotatedCapture) => box(rewrite(b), annotatedCapture) + case Pure.Select(target, field, annotatedType) => Pure.Select(rewrite(target), field, annotatedType) + case Pure.Box(b, annotatedCapture) => Pure.Box(rewrite(b), annotatedCapture) } def rewrite(e: Expr)(using StaticArgumentsContext): Expr = e match { - case DirectApp(b, targs, vargs, bargs) => directApp(rewrite(b), targs, vargs.map(rewrite), bargs.map(rewrite)) + case DirectApp(b, targs, vargs, bargs) => DirectApp(rewrite(b), targs, vargs.map(rewrite), bargs.map(rewrite)) // congruences - case Run(s) => run(rewrite(s)) + case Run(s) => Run(rewrite(s)) case pure: Pure => rewrite(pure) } diff --git a/effekt/shared/src/main/scala/effekt/generator/chez/ChezScheme.scala b/effekt/shared/src/main/scala/effekt/generator/chez/ChezScheme.scala index f83e0cec9..e91f2b568 100644 --- a/effekt/shared/src/main/scala/effekt/generator/chez/ChezScheme.scala +++ b/effekt/shared/src/main/scala/effekt/generator/chez/ChezScheme.scala @@ -3,7 +3,8 @@ package generator package chez import effekt.context.Context -import effekt.symbols.{Module, Symbol} +import effekt.core.optimizer.Optimizer +import effekt.symbols.{ Module, Symbol } import effekt.util.messages.ErrorReporter import kiama.output.PrettyPrinterTypes.Document import kiama.util.Source @@ -52,7 +53,7 @@ trait ChezScheme extends Compiler[String] { // ------------------------ // Source => Core => Chez lazy val Compile = - allToCore(Core) andThen Aggregate andThen core.Optimizer andThen Chez map { case (main, expr) => + allToCore(Core) andThen Aggregate andThen Optimizer andThen Chez map { case (main, expr) => (Map(main -> pretty(expr).layout), main) } diff --git a/effekt/shared/src/main/scala/effekt/generator/chez/Transformer.scala b/effekt/shared/src/main/scala/effekt/generator/chez/Transformer.scala index ab983ef7b..968b9d707 100644 --- a/effekt/shared/src/main/scala/effekt/generator/chez/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/generator/chez/Transformer.scala @@ -117,9 +117,11 @@ trait Transformer { // currently bidirectional handlers are not supported case Resume(k, Return(expr)) => chez.Call(toChez(k), List(toChez(expr))) + case Resume(k, other) => sys error s"Not supported yet: ${util.show(stmt)}" + case Region(body) => chez.Builtin("with-region", toChez(body)) - case other => chez.Let(Nil, toChez(other)) + case s: Scope => chez.Let(Nil, toChez(s)) } def toChez(decl: core.Declaration): List[chez.Def] = decl match { diff --git a/effekt/shared/src/main/scala/effekt/generator/js/JavaScript.scala b/effekt/shared/src/main/scala/effekt/generator/js/JavaScript.scala index ded31be25..cc97dcbb1 100644 --- a/effekt/shared/src/main/scala/effekt/generator/js/JavaScript.scala +++ b/effekt/shared/src/main/scala/effekt/generator/js/JavaScript.scala @@ -4,6 +4,7 @@ package js import effekt.PhaseResult.CoreTransformed import effekt.context.Context +import effekt.core.optimizer.{ DropBindings, Optimizer } import kiama.output.PrettyPrinterTypes.Document import kiama.util.Source @@ -41,7 +42,7 @@ class JavaScript(additionalFeatureFlags: List[String] = Nil) extends Compiler[St Frontend andThen Middleend } - lazy val Optimized = allToCore(Core) andThen Aggregate andThen core.Optimizer map { + lazy val Optimized = allToCore(Core) andThen Aggregate andThen Optimizer andThen DropBindings map { case input @ CoreTransformed(source, tree, mod, core) => val mainSymbol = Context.checkMain(mod) val mainFile = path(mod) diff --git a/effekt/shared/src/main/scala/effekt/generator/llvm/LLVM.scala b/effekt/shared/src/main/scala/effekt/generator/llvm/LLVM.scala index 774f0d73b..219be5c07 100644 --- a/effekt/shared/src/main/scala/effekt/generator/llvm/LLVM.scala +++ b/effekt/shared/src/main/scala/effekt/generator/llvm/LLVM.scala @@ -3,8 +3,9 @@ package generator package llvm import effekt.context.Context +import effekt.core.optimizer import effekt.machine -import kiama.output.PrettyPrinterTypes.{Document, emptyLinks} +import kiama.output.PrettyPrinterTypes.{ Document, emptyLinks } import kiama.util.Source @@ -38,7 +39,7 @@ class LLVM extends Compiler[String] { // The Compilation Pipeline // ------------------------ // Source => Core => Machine => LLVM - lazy val Compile = allToCore(Core) andThen Aggregate andThen core.PolymorphismBoxing andThen core.Optimizer andThen Machine map { + lazy val Compile = allToCore(Core) andThen Aggregate andThen core.PolymorphismBoxing andThen optimizer.Optimizer andThen Machine map { case (mod, main, prog) => (mod, llvm.Transformer.transform(prog)) } @@ -51,7 +52,7 @@ class LLVM extends Compiler[String] { // ----------------------------------- object steps { // intermediate steps for VSCode - val afterCore = allToCore(Core) andThen Aggregate andThen core.PolymorphismBoxing andThen core.Optimizer + val afterCore = allToCore(Core) andThen Aggregate andThen core.PolymorphismBoxing andThen optimizer.Optimizer val afterMachine = afterCore andThen Machine map { case (mod, main, prog) => prog } val afterLLVM = afterMachine map { case machine.Program(decls, prog) => diff --git a/effekt/shared/src/main/scala/effekt/machine/Transformer.scala b/effekt/shared/src/main/scala/effekt/machine/Transformer.scala index 34cd60ae5..0e010bd25 100644 --- a/effekt/shared/src/main/scala/effekt/machine/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/machine/Transformer.scala @@ -6,6 +6,8 @@ import effekt.core.{ Block, DeclarationContext, Definition, Id, given } import effekt.symbols.{ Symbol, TermSymbol } import effekt.symbols.builtins.TState import effekt.util.messages.ErrorReporter +import effekt.symbols.ErrorMessageInterpolator + object Transformer { @@ -121,16 +123,13 @@ object Transformer { noteDefinition(id, free, params) } - case Definition.Def(id, core.Unbox(_)) => - // TODO deal with this case - () + case Definition.Def(id, b @ core.Unbox(_)) => + noteParameter(id, b.tpe) case Definition.Let(_, _, _) => () } - - // (2) Actually translate the definitions definitions.foldRight(transform(rest)) { case (core.Definition.Let(id, tpe, binding), rest) => @@ -149,9 +148,10 @@ object Transformer { case (core.Definition.Def(id, core.BlockVar(alias, tpe, _)), rest) => Def(transformLabel(id), Jump(transformLabel(alias)), rest) - case (d @ core.Definition.Def(_, _: core.Unbox), rest) => - // TODO deal with this case by substitution - ErrorReporter.abort(s"block definition: $d") + case (core.Definition.Def(id, core.Unbox(pure)), rest) => + transform(pure).run { boxed => + ForeignCall(Variable(transform(id), Type.Negative()), "unbox", List(boxed), rest) + } } case core.Return(expr) => @@ -168,7 +168,7 @@ object Transformer { transform(vargs, bargs).run { (values, blocks) => callee match { case Block.BlockVar(id, annotatedTpe, annotatedCapt) => - BPC.info.getOrElse(id, sys.error(s"Cannot find block info for ${id}.\n${BPC.info}")) match { + BPC.info.getOrElse(id, sys.error(pp"In ${stmt}. Cannot find block info for ${id}: ${annotatedTpe}.\n${BPC.info}")) match { // Unknown Jump to function case BlockInfo.Parameter(tpe: core.BlockType.Function) => Invoke(Variable(transform(id), transform(tpe)), builtins.Apply, values ++ blocks) @@ -182,13 +182,18 @@ object Transformer { } case Block.Unbox(pure) => - transform(pure).run { callee => Invoke(callee, builtins.Apply, values ++ blocks) } + transform(pure).run { boxedCallee => + val callee = Variable(freshName(boxedCallee.name), Type.Negative()) + + ForeignCall(callee, "unbox", List(boxedCallee), + Invoke(callee, builtins.Apply, values ++ blocks)) + } case Block.New(impl) => ErrorReporter.panic("Applying an object") case Block.BlockLit(tparams, cparams, vparams, bparams, body) => - ErrorReporter.panic("Call to block literal should have been reduced") + ErrorReporter.panic(pp"Call to block literal should have been reduced: ${stmt}") } } @@ -202,7 +207,12 @@ object Transformer { Invoke(Variable(transform(id), transform(tpe)), opTag, values ++ blocks) case Block.Unbox(pure) => - transform(pure).run { callee => Invoke(callee, opTag, values ++ blocks) } + transform(pure).run { boxedCallee => + val callee = Variable(freshName(boxedCallee.name), Type.Negative()) + + ForeignCall(callee, "unbox", List(boxedCallee), + Invoke(callee, opTag, values ++ blocks)) + } case Block.New(impl) => ErrorReporter.panic("Method call to known object should have been reduced") @@ -451,7 +461,12 @@ object Transformer { } case core.Box(block, annot) => - transformBlockArg(block) + transformBlockArg(block).flatMap { unboxed => + Binding { k => + val boxed = Variable(freshName(unboxed.name), Type.Positive()) + ForeignCall(boxed, "box", List(unboxed), k(boxed)) + } + } case _ => ErrorReporter.abort(s"Unsupported expression: $expr") @@ -485,7 +500,7 @@ object Transformer { def transform(tpe: core.ValueType)(using ErrorReporter): Type = tpe match { case core.ValueType.Var(name) => Positive() // assume all value parameters are data - case core.ValueType.Boxed(tpe, capt) => Negative() + case core.ValueType.Boxed(tpe, capt) => Positive() case core.Type.TUnit => builtins.UnitType case core.Type.TInt => Type.Int() case core.Type.TChar => Type.Int() @@ -566,7 +581,7 @@ object Transformer { BPC.globals += (id -> Label(transform(id), Nil)) def getBlockInfo(id: Id)(using BPC: BlocksParamsContext): BlockInfo = - BPC.info.getOrElse(id, sys error s"No block info for ${id}") + BPC.info.getOrElse(id, sys error s"No block info for ${util.show(id)}") def getDefinition(id: Id)(using BPC: BlocksParamsContext): BlockInfo.Definition = getBlockInfo(id) match { case d : BlockInfo.Definition => d diff --git a/effekt/shared/src/main/scala/effekt/util/Debug.scala b/effekt/shared/src/main/scala/effekt/util/Debug.scala index ca9189ef3..f58d5d6ca 100644 --- a/effekt/shared/src/main/scala/effekt/util/Debug.scala +++ b/effekt/shared/src/main/scala/effekt/util/Debug.scala @@ -4,7 +4,7 @@ package util import effekt.symbols.TypePrinter -lazy val showGeneric: PartialFunction[Any, String] = { +val showGeneric: PartialFunction[Any, String] = { case l: List[_] => l.map(show).mkString("List(", ", ", ")") case o: Option[_] => @@ -12,7 +12,7 @@ lazy val showGeneric: PartialFunction[Any, String] = { case other => other.toString } -lazy val show: PartialFunction[Any, String] = +val show: PartialFunction[Any, String] = TypePrinter.show orElse core.PrettyPrinter.show orElse generator.js.PrettyPrinter.show orElse diff --git a/examples/benchmarks/are_we_fast_yet/queens.effekt b/examples/benchmarks/are_we_fast_yet/queens.effekt index 497ef8496..7ab35a427 100644 --- a/examples/benchmarks/are_we_fast_yet/queens.effekt +++ b/examples/benchmarks/are_we_fast_yet/queens.effekt @@ -49,4 +49,3 @@ def run(n: Int) = { } def main() = benchmark(8){run} - diff --git a/examples/benchmarks/effect_handlers_bench/parsing_dollars.effekt b/examples/benchmarks/effect_handlers_bench/parsing_dollars.effekt index 77ce1aaf1..11a662a89 100644 --- a/examples/benchmarks/effect_handlers_bench/parsing_dollars.effekt +++ b/examples/benchmarks/effect_handlers_bench/parsing_dollars.effekt @@ -28,7 +28,7 @@ def parse(a: Int): Unit / {Read, Emit, Stop} = { do Stop() } } - + def sum { action: () => Unit / Emit } = { var s = 0; try { diff --git a/examples/benchmarks/effect_handlers_bench/tree_explore.effekt b/examples/benchmarks/effect_handlers_bench/tree_explore.effekt index 6611050e8..fba8f712e 100644 --- a/examples/benchmarks/effect_handlers_bench/tree_explore.effekt +++ b/examples/benchmarks/effect_handlers_bench/tree_explore.effekt @@ -65,4 +65,3 @@ def run(n: Int) = { } def main() = benchmark(5){run} - diff --git a/libraries/common/io.effekt b/libraries/common/io.effekt index af47633bd..c3ab21adf 100644 --- a/libraries/common/io.effekt +++ b/libraries/common/io.effekt @@ -16,7 +16,8 @@ extern async def spawn(task: Task[Unit]): Unit = js "$effekt.capture(k => { setTimeout(() => k($effekt.unit), 0); return $effekt.run(${task}) })" llvm """ call void @c_yield(%Stack %stack) - call void @run(%Neg ${task}) + %unboxed = call ccc %Neg @unbox(%Pos ${task}) + call void @run(%Neg %unboxed) ret void """ diff --git a/libraries/llvm/rts.ll b/libraries/llvm/rts.ll index f69fd3dac..cd3fb0d8a 100644 --- a/libraries/llvm/rts.ll +++ b/libraries/llvm/rts.ll @@ -114,6 +114,26 @@ declare void @exit(i64) declare void @llvm.assume(i1) +; Boxing (externs functions, hence ccc) +define ccc %Pos @box(%Neg %input) { + %vtable = extractvalue %Neg %input, 0 + %heap_obj = extractvalue %Neg %input, 1 + %vtable_as_int = ptrtoint ptr %vtable to i64 + %pos_result = insertvalue %Pos undef, i64 %vtable_as_int, 0 + %pos_result_with_heap = insertvalue %Pos %pos_result, ptr %heap_obj, 1 + ret %Pos %pos_result_with_heap +} + +define ccc %Neg @unbox(%Pos %input) { + %tag = extractvalue %Pos %input, 0 + %heap_obj = extractvalue %Pos %input, 1 + %vtable = inttoptr i64 %tag to ptr + %neg_result = insertvalue %Neg undef, ptr %vtable, 0 + %neg_result_with_heap = insertvalue %Neg %neg_result, ptr %heap_obj, 1 + ret %Neg %neg_result_with_heap +} + + ; Prompts define private %Prompt @currentPrompt(%Stack %stack) { From 3fec7a068b5fbeaeb0f0e28b2ec7c17bcdee61a5 Mon Sep 17 00:00:00 2001 From: "effekt-updater[bot]" <181701480+effekt-updater[bot]@users.noreply.github.com> Date: Mon, 6 Jan 2025 03:37:06 +0000 Subject: [PATCH 14/31] Bump version to 0.16.0 --- package.json | 2 +- pom.xml | 2 +- project/EffektVersion.scala | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/package.json b/package.json index a735fe933..a4ee9d884 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "@effekt-lang/effekt", "author": "Jonathan BrachthΓ€user", - "version": "0.15.0", + "version": "0.16.0", "repository": { "type": "git", "url": "git+https://github.com/effekt-lang/effekt.git" diff --git a/pom.xml b/pom.xml index 28aa9d724..b0cf6ccaf 100644 --- a/pom.xml +++ b/pom.xml @@ -7,7 +7,7 @@ de.bstudios Effekt Effekt - 0.15.0 + 0.16.0 4.0.0 diff --git a/project/EffektVersion.scala b/project/EffektVersion.scala index 95bdf67eb..f0ede4db4 100644 --- a/project/EffektVersion.scala +++ b/project/EffektVersion.scala @@ -1,4 +1,4 @@ // Don't change this file without changing the CI too! import sbt.* import sbt.Keys.* -object EffektVersion { lazy val effektVersion = "0.15.0" } +object EffektVersion { lazy val effektVersion = "0.16.0" } From 82c76110bcc07e089548c26060d25422b5bec799 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jonathan=20Immanuel=20Brachth=C3=A4user?= Date: Tue, 7 Jan 2025 13:49:12 +0100 Subject: [PATCH 15/31] Refactor core (remove Run, Scope, Params...) (#759) This PR - deletes `Run` from core - removes `Scope` from core and adds `Def` and `Let` to `Stmt`, instead. - restricts callees of `DirectApp` and `PureApp` to be variables - cleans up `PolymorphismBoxing` a bit - removes the `Params` abstraction from core Dropping `Run` enables #713 which currently stack overflows, due to using the host language stack. --- .../effekt/core/LambdaLiftingTests.scala | 39 -- .../effekt/core/PolymorphismBoxingTests.scala | 12 +- .../test/scala/effekt/core/SizeTests.scala | 4 +- .../src/main/scala/effekt/Compiler.scala | 2 +- .../scala/effekt/core/LambdaLifting.scala | 245 ---------- .../scala/effekt/core/MakeStackSafe.scala | 60 --- .../src/main/scala/effekt/core/Parser.scala | 77 ++-- .../effekt/core/PolymorphismBoxing.scala | 419 ++++++++---------- .../scala/effekt/core/PrettyPrinter.scala | 32 +- .../main/scala/effekt/core/Recursive.scala | 38 +- .../src/main/scala/effekt/core/Renamer.scala | 19 +- .../main/scala/effekt/core/Transformer.scala | 82 +--- .../src/main/scala/effekt/core/Tree.scala | 251 +++++------ .../src/main/scala/effekt/core/Type.scala | 13 +- .../core/optimizer/BindSubexpressions.scala | 94 ++-- .../effekt/core/optimizer/Deadcode.scala | 10 +- .../effekt/core/optimizer/DropBindings.scala | 15 +- .../effekt/core/optimizer/Normalizer.scala | 132 ++---- .../effekt/core/optimizer/Reachable.scala | 67 +-- .../optimizer/RemoveTailResumptions.scala | 8 +- .../core/optimizer/StaticArguments.scala | 39 +- .../main/scala/effekt/cps/Transformer.scala | 59 +-- .../src/main/scala/effekt/cps/Tree.scala | 7 +- .../effekt/generator/chez/Transformer.scala | 59 ++- .../effekt/generator/js/Transformer.scala | 2 +- .../effekt/generator/js/TransformerCps.scala | 4 +- .../scala/effekt/machine/Transformer.scala | 142 +++--- 27 files changed, 662 insertions(+), 1269 deletions(-) delete mode 100644 effekt/jvm/src/test/scala/effekt/core/LambdaLiftingTests.scala delete mode 100644 effekt/shared/src/main/scala/effekt/core/LambdaLifting.scala delete mode 100644 effekt/shared/src/main/scala/effekt/core/MakeStackSafe.scala diff --git a/effekt/jvm/src/test/scala/effekt/core/LambdaLiftingTests.scala b/effekt/jvm/src/test/scala/effekt/core/LambdaLiftingTests.scala deleted file mode 100644 index 841a9e6b4..000000000 --- a/effekt/jvm/src/test/scala/effekt/core/LambdaLiftingTests.scala +++ /dev/null @@ -1,39 +0,0 @@ -package effekt.core - - -class LambdaLiftingTests extends CorePhaseTests(LambdaLifting) { - - test("toplevel functions stay unchanged"){ - val from = - """module main - | - |def id = { ['A](a: 'A) => return a: 'A } - |""".stripMargin - assertTransformsTo(from, from) - } - - test("local functions are lifted out"){ - val from = - """module main - | - |def outer = { () => - | def local = { () => return 42 } - | val res = (local: () => Int @ {})(); - | return res:Int - |} - |""".stripMargin - - val to = - """module main - | - |def outer = { () => - | val res = (local: () => Int @ {})(); - | return res:Int - |} - | - |def local = { () => return 42 } - |""".stripMargin - assertTransformsTo(from, to) - } - -} diff --git a/effekt/jvm/src/test/scala/effekt/core/PolymorphismBoxingTests.scala b/effekt/jvm/src/test/scala/effekt/core/PolymorphismBoxingTests.scala index bf3030db9..6894395a6 100644 --- a/effekt/jvm/src/test/scala/effekt/core/PolymorphismBoxingTests.scala +++ b/effekt/jvm/src/test/scala/effekt/core/PolymorphismBoxingTests.scala @@ -3,7 +3,7 @@ package core import effekt.{core, source, symbols} import effekt.context.Context -import effekt.core.{Block, Definition, DirectApp, PolymorphismBoxing, Pure, Run, Stmt} +import effekt.core.{Block, DirectApp, PolymorphismBoxing, Pure, Stmt} import effekt.source.{IdDef, Include} import effekt.util.messages import effekt.util.messages.DebugMessaging @@ -125,13 +125,9 @@ class PolymorphismBoxingTests extends AbstractPolymorphismBoxingTests { | |def id = { ['A](a: 'A) => return a: 'A } |def idInt = { (x: Int) => - | { - | let res = run { - | let boxedRes = !(id: ['A]('A) => 'A @ {})[BoxedInt]((boxInt: (Int) => BoxedInt @ {})(x: Int)) - | return (unboxInt: (BoxedInt) => Int @ {})(boxedRes:BoxedInt) - | } - | return res: Int - | } + | let boxed = !(id: ['A]('A) => 'A @ {})[BoxedInt]((boxInt: (Int) => BoxedInt @ {})(x: Int)) + | let unboxed = (unboxInt: (BoxedInt) => Int @ {})(boxed:BoxedInt) + | return unboxed: Int |} |""".stripMargin assertTransformsTo(from,to) diff --git a/effekt/jvm/src/test/scala/effekt/core/SizeTests.scala b/effekt/jvm/src/test/scala/effekt/core/SizeTests.scala index 4a8eef04f..81a939369 100644 --- a/effekt/jvm/src/test/scala/effekt/core/SizeTests.scala +++ b/effekt/jvm/src/test/scala/effekt/core/SizeTests.scala @@ -8,7 +8,7 @@ class SizeTests extends CoreTests { } test("Small program"){ - assertSized(7) { + assertSized(6) { """module main | |def foo = { () => @@ -19,7 +19,7 @@ class SizeTests extends CoreTests { } test("Nested definitions") { - assertSized(18) { + assertSized(15) { """ module main | | def bar = { () => return 1 } diff --git a/effekt/shared/src/main/scala/effekt/Compiler.scala b/effekt/shared/src/main/scala/effekt/Compiler.scala index d7062642c..a583f6554 100644 --- a/effekt/shared/src/main/scala/effekt/Compiler.scala +++ b/effekt/shared/src/main/scala/effekt/Compiler.scala @@ -284,7 +284,7 @@ trait Compiler[Executable] { // collect all information var declarations: List[core.Declaration] = Nil var externs: List[core.Extern] = Nil - var definitions: List[core.Definition] = Nil + var definitions: List[core.Toplevel] = Nil var exports: List[symbols.Symbol] = Nil (dependencies :+ main).foreach { module => diff --git a/effekt/shared/src/main/scala/effekt/core/LambdaLifting.scala b/effekt/shared/src/main/scala/effekt/core/LambdaLifting.scala deleted file mode 100644 index 27afab523..000000000 --- a/effekt/shared/src/main/scala/effekt/core/LambdaLifting.scala +++ /dev/null @@ -1,245 +0,0 @@ -package effekt -package core - -import effekt.context.Context - -import scala.collection.mutable -import effekt.core.Variables -import effekt.core.Variables.{ all, bound, free } - -class LambdaLifting(m: core.ModuleDecl)(using Context) extends core.Tree.Rewrite { - - val locals = Locals(m) - - /** - * fixes the order of free variables, can vary from compilation to compilation - */ - case class Info(values: List[Variable.Value], blocks: List[Variable.Block]) { - def valueParams: List[core.ValueParam] = values.map { case Variable.Value(id, tpe) => core.ValueParam(id, tpe) } - def blockParams: List[core.BlockParam] = blocks.map { case Variable.Block(id, tpe, capt) => core.BlockParam(id, tpe, capt) } - def captureParams: List[core.Capture] = blocks.map { - case Variable.Block(id, tpe, cs) if cs.size == 1 => cs.head - case Variable.Block(id, tpe, cs) => Context.panic(s"Since we only close over block parameters, the capture set should be a single variable (but got ${cs} for ${id})") - } - - def valueArgs = values.map { case Variable.Value(id, tpe) => core.ValueVar(id, tpe) } - def blockArgs = blocks.map { case Variable.Block(id, tpe, capt) => core.BlockVar(id, tpe, capt) } - def captureArgs = blocks.map { case Variable.Block(id, tpe, cs) => cs } - } - val infos: Map[Id, Info] = locals.transitiveClosure.map { - case (id, vars) => (id, Info( - vars.toList.collect { case x: Variable.Value => x }, - vars.toList.collect { case f: Variable.Block => f } - )) - }.toMap - val lifted: mutable.ListBuffer[core.Definition] = mutable.ListBuffer.empty - - // only needs adaptation if it is a closure - def needsCallsiteAdaptation(id: Id) = infos.get(id) match { - case Some(vars) => vars != Variables.empty - case None => false - } - - // we adapt the type of the reference since now it closes over less variables but receives more as arguments - // e.g. (Int) => Unit at {io, f} ===> (Int, f: Exc) => Unit at {io} - def adaptReference(b: BlockVar): BlockVar = b match - case b if !needsCallsiteAdaptation(b.id) => b - case BlockVar(id, BlockType.Function(tps, cps, vps, bps, ret), annotatedCapt) => - val info = infos(id) - val additionalValues = info.values.map { x => x.tpe } - val (additionalCaptures, additionalBlocks, removedCaptures) = info.blocks.map { - case Variable.Block(id, tpe, capt) => (id, tpe, capt) - }.unzip3 - val newType = BlockType.Function(tps, cps ++ additionalCaptures, vps ++ additionalValues, bps ++ additionalBlocks, ret) - // TODO what if the block parameters have been renamed somewhere---subtracting from capture won't help then. - val newCapture = annotatedCapt -- removedCaptures.flatten - BlockVar(id, newType, newCapture) - - case other => Context.panic("Cannot lambda lift non-functions.") - - override def stmt = { - case core.Scope(defs, body) => - MaybeScope(defs.flatMap { - // we lift named local definitions to the toplevel - case Definition.Def(id, BlockLit(tparams, cparams, vparams, bparams, body)) => - lifted.append(Definition.Def(id, - Renamer.rename(BlockLit(tparams, - // Here we add cparams for the closed over bparams - cparams ++ infos(id).captureParams, - vparams ++ infos(id).valueParams, - bparams ++ infos(id).blockParams, - rewrite(body))))) - Nil - case other => List(rewrite(other)) - }, rewrite(body)) - - case core.App(b: BlockVar, targs, vargs, bargs) if needsCallsiteAdaptation(b.id) => - core.App(adaptReference(b), targs, vargs.map(rewrite) ++ infos(b.id).valueArgs, bargs.map(rewrite) ++ infos(b.id).blockArgs) - } - - override def block = { - // Here we now need to eta expand - // e.g. f : (Int) => Unit @ {io,exc} ===> { (n) => f(n, exc) } - // the type of f after transformation is `(Int, Exc) => Unit @ {io}` - case f @ core.BlockVar(id, core.BlockType.Function(tps, cps, vps, bps, res), capt) if needsCallsiteAdaptation(id) => - val vparams: List[core.ValueParam] = vps map { tpe => core.ValueParam(Id("x"), tpe) } - val bparams: List[core.BlockParam] = (cps zip bps) map { case (capt, tpe) => core.BlockParam(Id("f"), tpe, Set(capt)) } - - val targs = tps map { tpe => core.ValueType.Var(tpe) } - val vargs = vparams.map { p => core.ValueVar(p.id, p.tpe) } ++ infos(id).valueArgs - val bargs = (bparams zip cps).map { case (p, c) => core.BlockVar(p.id, p.tpe, Set(c)) } ++ infos(id).blockArgs - core.BlockLit(tps, cps, vparams, bparams, core.App(adaptReference(f), targs, vargs, bargs)) - } - - override def expr = { - case core.DirectApp(b: BlockVar, targs, vargs, bargs) if needsCallsiteAdaptation(b.id) => - core.DirectApp(b, targs, vargs.map(rewrite) ++ infos(b.id).valueArgs, bargs.map(rewrite) ++ infos(b.id).blockArgs) - case core.PureApp(b: BlockVar, targs, vargs) if needsCallsiteAdaptation(b.id) => - core.PureApp(b, targs, vargs.map(rewrite) ++ infos(b.id).valueArgs) - } -} - -object LambdaLifting extends Phase[CoreTransformed, CoreTransformed] { - - val phaseName: String = "core-lambdalifting" - - def run(input: CoreTransformed)(using Context): Option[CoreTransformed] = - input match { - case CoreTransformed(source, tree, mod, core) => - val lifted = Context.timed(phaseName, source.name) { lift(core) } - Some(CoreTransformed(source, tree, mod, lifted)) - } - - def lift(m: core.ModuleDecl)(using Context): core.ModuleDecl = - val lifting = new LambdaLifting(m) - val transformed = lifting.rewrite(m) - transformed.copy(definitions = transformed.definitions ++ lifting.lifted) -} - -/** - * Free variable computation that annotates Val and Def trees with the free variables of - * their continuation / body, respectively. - * - * Use like: - * - * val locals = new Locals(mod); - * ... - * locals(myValDef) // Variables(Set(), Set(f17)) - * - * WARNING: the mapping is performed by object identity, so rewriting the tree looses the annotations. - * WARNING: since the local-context is lost, do NOT use it by querying on demand (e.g. `locals.query(myTree)`) - */ -class Locals(mod: ModuleDecl)(using Context) extends core.Tree.Query[Variables, Variables] { - - // DB - // -- - import effekt.context.{Annotations, Annotation} - - private val LocallyFree = Annotation[core.Tree, core.Variables]( - "LocallyFree", - "the free variables of the tree, only considering local and not toplevel definitions" - ) - - private val db = Annotations.empty - - def apply(t: core.Val | core.Definition): core.Variables = db.apply(LocallyFree, t) - - // Monoid - // ------ - def empty = Variables.empty - def combine = _ ++ _ - - // Scoping - // ------- - def freeBlock(id: Id)(using L: Variables): Variables = L.filter(v => v.id == id) - def freeValue(id: Id)(using L: Variables): Variables = L.filter(v => v.id == id) - def binding(bound: Variables)(prog: Variables ?=> Variables)(using L: Variables): Variables = - prog(using L ++ bound) -- bound - - override def pure(using Variables) = { - case core.ValueVar(id, annotatedType) => freeValue(id) - } - - override def block(using Variables) = { - case core.BlockVar(id, annotatedTpe, annotatedCapt) => freeBlock(id) - case core.BlockLit(tparams, cparams, vparams, bparams, body) => - binding(all(vparams, bound) ++ all(bparams, bound)) { query(body) } - } - - override def operation(using Variables) = { - case core.Operation(name, tparams, cparams, vparams, bparams, body) => - binding(all(vparams, bound) ++ all(bparams, bound)) { query(body) } - } - - override def defn(using Variables) = { - case d @ core.Definition.Def(id, block) => - val freeInDefinition = binding(bound(d)) { query(block) } - // we annotate free variables for each definition (Def) - db.update(LocallyFree, d, freeInDefinition) - freeInDefinition - } - - override def stmt(using Variables) = { - case Stmt.Scope(defs, body) => - var stillFree = Variables.empty - var boundSoFar = Variables.empty - defs.foreach { d => - boundSoFar = boundSoFar ++ bound(d) - stillFree = stillFree ++ binding(boundSoFar) { query(d) } - } - stillFree ++ binding(boundSoFar) { query(body) } - - case d @ Stmt.Val(id, tpe, rhs, body) => - val bound = Variables.value(id, rhs.tpe) - query(rhs) ++ binding(bound) { - // we annotate the free variables of the continuation - val freeInBody = query(body) - db.update(LocallyFree, d, freeInBody -- bound) - freeInBody - } - - case core.Alloc(id, init, region, body) => - val bound = Variables.block(id, Type.TState(init.tpe), Set(region)) - query(init) ++ freeBlock(region) ++ binding(bound) { query(body) } - case core.Var(id, init, capture, body) => - val bound = Variables.block(id, Type.TState(init.tpe), Set(capture)) - query(init) ++ binding(bound) { query(body) } - case core.Get(id, annotatedCapt, annotatedTpe) => freeBlock(id) - case core.Put(id, annotatedCapt, value) => freeBlock(id) ++ free(value) - } - - // Initialize - // ---------- - mod.definitions.foreach(d => query(d)(using Variables.empty)) - - // maps block ids to their transitive closure - val transitiveClosure: mutable.Map[Id, Variables] = mutable.Map.empty - - // compute transitive closure - val freeVariablesOfDefs = db.annotationsAt(LocallyFree).collect { - case (Annotations.Key(core.Definition.Def(id, b: core.BlockLit)), vars) => id -> vars - } - - // saturate free variables transitively - def resolveFreeVariables(vars: Variables): Variables = - vars.flatMap { - case x: Variable.Value => Variables(Set(x)) - case f: Variable.Block => resolve(f.id).getOrElse(Variables(Set(f))) - } - - def resolve(id: Id): Option[Variables] = - transitiveClosure.get(id) match { - case Some(value) => Some(value) - case None => - freeVariablesOfDefs.get(id).map { before => - transitiveClosure.update(id, Variables.empty) - val result = resolveFreeVariables(before) - transitiveClosure.update(id, result) - result - } - } - - - freeVariablesOfDefs.keySet.foreach { resolve } - -} diff --git a/effekt/shared/src/main/scala/effekt/core/MakeStackSafe.scala b/effekt/shared/src/main/scala/effekt/core/MakeStackSafe.scala deleted file mode 100644 index 81e8cb333..000000000 --- a/effekt/shared/src/main/scala/effekt/core/MakeStackSafe.scala +++ /dev/null @@ -1,60 +0,0 @@ -package effekt -package core - -import effekt.context.Context -import effekt.symbols.{ TmpBlock, TmpValue } -import effekt.Phase -import effekt.PhaseResult.CoreTransformed -import effekt.core.Block.BlockLit - -/** - * [[Phase]] on [[CoreTransformed]] to make programs stack safe on platforms - * that do not support TCO. - * - * This is necessary for backends like JS, where we should use the monadic trampoline - * for recursive functions. - */ -object MakeStackSafe extends Phase[CoreTransformed, CoreTransformed] { - - override val phaseName: String = "stacksafe" - - override def run(input: CoreTransformed)(using Context): Option[CoreTransformed] = input match { - case CoreTransformed(source, tree, mod, core) => - val safe = Context.timed(phaseName, source.name) { stacksafe.rewrite(core) } - Some(CoreTransformed(source, tree, mod, safe)) - } - - object stacksafe extends Tree.Rewrite { - - var scopes: List[Id] = Nil - def within[T](id: Id)(f: => T) : T = { - val before = scopes - scopes = id :: before - val result = f - scopes = before - result - } - def isRecursive(id: Id): Boolean = scopes contains id - def clear() = scopes = Nil - - override def stmt: PartialFunction[Stmt, Stmt] = { - case Stmt.Val(id, tpe, binding, body) => - Stmt.Val(id, tpe, rewrite(binding), { - // stop transformation under val binders, they already perform trampolining - clear(); - rewrite(body) - }) - - case Stmt.App(x : BlockVar, targs, vargs, bargs) if isRecursive(x.id) => - thunk { Stmt.App(x, targs, vargs.map(rewrite), bargs.map(rewrite)) } - } - - override def defn: PartialFunction[Definition, Definition] = { - case Definition.Def(id, b : Block.BlockLit) => - Definition.Def(id, within(id) { rewrite(b) }) - } - } - - // [[ s ]] = val tmp = return (); s - def thunk(s: Stmt): Stmt = Stmt.Val(TmpValue(), core.Type.TUnit, Stmt.Return(Literal((), core.Type.TUnit)), s) -} diff --git a/effekt/shared/src/main/scala/effekt/core/Parser.scala b/effekt/shared/src/main/scala/effekt/core/Parser.scala index 7d80f3607..b2175012a 100644 --- a/effekt/shared/src/main/scala/effekt/core/Parser.scala +++ b/effekt/shared/src/main/scala/effekt/core/Parser.scala @@ -1,7 +1,7 @@ package effekt package core -import effekt.core.Param.ValueParam +import effekt.core.ValueParam import effekt.source.{FeatureFlag, NoSource} import effekt.util.messages.{ DebugMessaging, ErrorReporter, ParseError } import kiama.parsing.{ Failure, Input, NoSuccess, ParseResult, Success } @@ -57,7 +57,7 @@ class CoreParsers(positions: Positions, names: Names) extends EffektLexers(posit many(includeDecl) ~ many(declaration) ~ many(externDecl) ~ - many(definition) ~ + many(toplevel) ~ many(exportDecl) ^^ ModuleDecl.apply lazy val includeDecl: P[String] = @@ -103,14 +103,14 @@ class CoreParsers(positions: Positions, names: Names) extends EffektLexers(posit // Definitions // ----------- - lazy val definition: P[Definition] = - ( `let` ~> id ~ maybeTypeAnnotation ~ (`=` ~/> expr) ^^ { - case (name ~ tpe ~ binding) => Definition.Let(name, tpe.getOrElse(binding.tpe), binding) + lazy val toplevel: P[Toplevel] = + ( `val` ~> id ~ maybeTypeAnnotation ~ (`=` ~/> stmt) ^^ { + case (name ~ tpe ~ binding) => Toplevel.Val(name, tpe.getOrElse(binding.tpe), binding) } - | `def` ~> id ~ (`=` ~/> block) ^^ Definition.Def.apply + | `def` ~> id ~ (`=` ~/> block) ^^ Toplevel.Def.apply | `def` ~> id ~ parameters ~ (`=` ~> stmt) ^^ { case name ~ (tparams, cparams, vparams, bparams) ~ body => - Definition.Def(name, BlockLit(tparams, cparams, vparams, bparams, body)) + Toplevel.Def(name, BlockLit(tparams, cparams, vparams, bparams, body)) } | failure("Expected a definition.") ) @@ -119,28 +119,34 @@ class CoreParsers(positions: Positions, names: Names) extends EffektLexers(posit // Statements // ---------- lazy val stmt: P[Stmt] = - ( `{` ~/> many(definition) ~ stmt <~ `}` ^^ Stmt.Scope.apply // curly braces induce scopes! + ( `{` ~/> stmts <~ `}` | `return` ~> pure ^^ Stmt.Return.apply - | `val` ~> id ~ maybeTypeAnnotation ~ (`=` ~> stmt) ~ (`;` ~> stmt) ^^ { - case id ~ tpe ~ binding ~ body => Stmt.Val(id, tpe.getOrElse(binding.tpe), binding, body) - } | block ~ (`.` ~> id ~ (`:` ~> blockType)).? ~ maybeTypeArgs ~ valueArgs ~ blockArgs ^^ { - case (recv ~ Some(method ~ tpe) ~ targs ~ vargs ~ bargs) => Invoke(recv, method, tpe, targs, vargs, bargs) - case (recv ~ None ~ targs ~ vargs ~ bargs) => App(recv, targs, vargs, bargs) - } + case (recv ~ Some(method ~ tpe) ~ targs ~ vargs ~ bargs) => Invoke(recv, method, tpe, targs, vargs, bargs) + case (recv ~ None ~ targs ~ vargs ~ bargs) => App(recv, targs, vargs, bargs) + } | (`if` ~> `(` ~/> pure <~ `)`) ~ stmt ~ (`else` ~> stmt) ^^ Stmt.If.apply | `region` ~> blockLit ^^ Stmt.Region.apply - | `var` ~> id ~ (`in` ~> id) ~ (`=` ~> pure) ~ (`;` ~> stmt) ^^ { case id ~ region ~ init ~ body => Alloc(id, init, region, body) } - | `var` ~> id ~ (`@` ~> id) ~ (`=` ~> pure) ~ (`;` ~> stmt) ^^ { case id ~ cap ~ init ~ body => Var(id, init, cap, body) } | `<>` ^^^ Hole() | (pure <~ `match`) ~/ (`{` ~> many(clause) <~ `}`) ~ (`else` ~> stmt).? ^^ Stmt.Match.apply ) lazy val stmts: P[Stmt] = - many(definition) ~ stmt ^^ { - case Nil ~ stmt => stmt - case defs ~ stmt => Stmt.Scope(defs, stmt) - } + ( `let` ~/> id ~ maybeTypeAnnotation ~ (`=` ~/> expr) ~ stmts ^^ { + case (name ~ tpe ~ binding ~ body) => Let(name, tpe.getOrElse(binding.tpe), binding, body) + } + | `def` ~> id ~ (`=` ~/> block) ~ stmts ^^ Stmt.Def.apply + | `def` ~> id ~ parameters ~ (`=` ~/> stmt) ~ stmts ^^ { + case name ~ (tparams, cparams, vparams, bparams) ~ body ~ rest => + Stmt.Def(name, BlockLit(tparams, cparams, vparams, bparams, body), rest) + } + | `val` ~> id ~ maybeTypeAnnotation ~ (`=` ~> stmt) ~ (`;` ~> stmts) ^^ { + case id ~ tpe ~ binding ~ body => Val(id, tpe.getOrElse(binding.tpe), binding, body) + } + | `var` ~> id ~ (`in` ~> id) ~ (`=` ~> pure) ~ (`;` ~> stmts) ^^ { case id ~ region ~ init ~ body => Alloc(id, init, region, body) } + | `var` ~> id ~ (`@` ~> id) ~ (`=` ~> pure) ~ (`;` ~> stmts) ^^ { case id ~ cap ~ init ~ body => Var(id, init, cap, body) } + | stmt + ) lazy val clause: P[(Id, BlockLit)] = (id <~ `:`) ~ blockLit ^^ { case id ~ cl => id -> cl } @@ -170,7 +176,7 @@ class CoreParsers(positions: Positions, names: Names) extends EffektLexers(posit | id ~ (`:` ~> valueType) ^^ Pure.ValueVar.apply | `box` ~> captures ~ block ^^ { case capt ~ block => Pure.Box(block, capt) } | `make` ~> dataType ~ id ~ valueArgs ^^ Pure.Make.apply - | block ~ maybeTypeArgs ~ valueArgs ^^ Pure.PureApp.apply + | maybeParensBlockVar ~ maybeTypeArgs ~ valueArgs ^^ Pure.PureApp.apply | failure("Expected a pure expression.") ) @@ -190,15 +196,14 @@ class CoreParsers(positions: Positions, names: Names) extends EffektLexers(posit // ----------- lazy val expr: P[Expr] = ( pure - | `run` ~> stmt ^^ Run.apply - | (`!` ~/> block) ~ maybeTypeArgs ~ valueArgs ~ blockArgs ^^ DirectApp.apply + | (`!` ~/> maybeParensBlockVar) ~ maybeTypeArgs ~ valueArgs ~ blockArgs ^^ DirectApp.apply ) // Blocks // ------ lazy val block: P[Block] = - ( id ~ (`:` ~> blockType) ~ (`@` ~> captures) ^^ Block.BlockVar.apply + ( blockVar | `unbox` ~> pure ^^ Block.Unbox.apply | `new` ~> implementation ^^ Block.New.apply | blockLit @@ -206,6 +211,16 @@ class CoreParsers(positions: Positions, names: Names) extends EffektLexers(posit | `(` ~> block <~ `)` ) + lazy val blockVar: P[Block.BlockVar] = + id ~ (`:` ~> blockType) ~ (`@` ~> captures) ^^ { + case (id ~ tpe ~ capt) => Block.BlockVar(id, tpe, capt) : Block.BlockVar + } + + lazy val maybeParensBlockVar: P[Block.BlockVar] = + ( `(` ~> blockVar <~ `)` + | blockVar + ) + lazy val blockLit: P[Block.BlockLit] = `{` ~> parameters ~ (`=>` ~/> stmts) <~ `}` ^^ { case (tparams, cparams, vparams, bparams) ~ body => @@ -220,13 +235,13 @@ class CoreParsers(positions: Positions, names: Names) extends EffektLexers(posit // Signatures // ------- // foo[Int, String](x: Int) { f@f2: Exc }: Int - lazy val signature: P[(Id, List[Id], List[Id], List[Param.ValueParam], List[Param.BlockParam], ValueType)] = + lazy val signature: P[(Id, List[Id], List[Id], List[ValueParam], List[BlockParam], ValueType)] = id ~ parameters ~ (`:` ~> valueType) ^^ { case name ~ (tparams, cparams, vparams, bparams) ~ result => (name, tparams, cparams, vparams, bparams, result) } - lazy val parameters: P[(List[Id], List[Id], List[Param.ValueParam], List[Param.BlockParam])] = + lazy val parameters: P[(List[Id], List[Id], List[ValueParam], List[BlockParam])] = maybeTypeParams ~ valueParams ~ many(trackedBlockParam) ^^ { case tparams ~ vparams ~ bcparams => val (cparams, bparams) = bcparams.unzip @@ -239,7 +254,7 @@ class CoreParsers(positions: Positions, names: Names) extends EffektLexers(posit lazy val valueParam: P[ValueParam] = - id ~ (`:` ~> valueType) ^^ { case id ~ tpe => Param.ValueParam(id, tpe): Param.ValueParam } + id ~ (`:` ~> valueType) ^^ { case id ~ tpe => ValueParam(id, tpe) } lazy val valueParams: P[List[ValueParam]] = `(` ~> manySep(valueParam, `,`) <~ `)` @@ -247,14 +262,14 @@ class CoreParsers(positions: Positions, names: Names) extends EffektLexers(posit // f@f2 : Exc lazy val trackedBlockParam: P[(Id, BlockParam)] = ( `{` ~> id ~ (`@` ~> id) ~ (`:` ~> blockType) <~ `}` ^^ { - case id ~ capt ~ tpe => capt -> (Param.BlockParam(id, tpe, Set(capt)): Param.BlockParam) + case id ~ capt ~ tpe => capt -> BlockParam(id, tpe, Set(capt)) } // abbreviation: f : Exc .= f@f : Exc | blockParam ^^ { p => p.id -> p } ) lazy val blockParam: P[BlockParam] = - `{` ~> id ~ (`:` ~> blockType) <~ `}` ^^ { case id ~ tpe => Param.BlockParam(id, tpe, Set(id)): Param.BlockParam } + `{` ~> id ~ (`:` ~> blockType) <~ `}` ^^ { case id ~ tpe => BlockParam(id, tpe, Set(id)) } // Types @@ -334,7 +349,7 @@ object CoreParsers { val parsers = CoreParsers(names) parsers.parseAll(parsers.stmt, input) - def definition(input: String, names: Names): ParseResult[Definition] = + def definition(input: String, names: Names): ParseResult[Toplevel] = val parsers = CoreParsers(names) - parsers.parseAll(parsers.definition, input) + parsers.parseAll(parsers.toplevel, input) } diff --git a/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala b/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala index 8030e7dff..553f4d7c8 100644 --- a/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala +++ b/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala @@ -3,10 +3,11 @@ package core import effekt.PhaseResult.CoreTransformed import effekt.context.Context +import effekt.core.PolymorphismBoxing.ValueCoercer.IdentityCoercer import effekt.symbols -import effekt.symbols.{TmpBlock, TmpValue} -import effekt.{CoreTransformed, Phase} -import effekt.symbols.builtins.{TBoolean, TByte, TChar, TDouble, TInt, TState, TUnit} +import effekt.symbols.{ TmpBlock, TmpValue } +import effekt.{ CoreTransformed, Phase } +import effekt.symbols.builtins.{ TBoolean, TByte, TChar, TDouble, TInt, TState, TUnit } import effekt.symbols.ErrorMessageInterpolator import scala.util.boundary @@ -146,13 +147,14 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { implicit val pctx: PContext = new PContext(core.declarations, core.externs) Context.module = mod val transformed = Context.timed(phaseName, source.name) { transform(core) } + Some(CoreTransformed(source, tree, mod, transformed)) } } def transform(decl: ModuleDecl)(using PContext): ModuleDecl = decl match { case ModuleDecl(path, includes, declarations, externs, definitions, exports) => - ModuleDecl(path, includes, declarations map transform, externs map transform, definitions flatMap transform, exports) + ModuleDecl(path, includes, declarations map transform, externs map transform, definitions map transform, exports) } def transform(declaration: Declaration)(using PContext): Declaration = declaration match { @@ -184,26 +186,17 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { case Extern.Include(ff, contents) => Extern.Include(ff, contents) } - def transform(valueParam: Param.ValueParam)(using PContext): Param.ValueParam = valueParam match { - case Param.ValueParam(id, tpe) => Param.ValueParam(id, transform(tpe)) + def transform(valueParam: ValueParam)(using PContext): ValueParam = valueParam match { + case ValueParam(id, tpe) => ValueParam(id, transform(tpe)) } - def transform(blockParam: Param.BlockParam)(using PContext): Param.BlockParam = blockParam match { - case Param.BlockParam(id, tpe, capt) => Param.BlockParam(id, transform(tpe), capt) + + def transform(blockParam: BlockParam)(using PContext): BlockParam = blockParam match { + case BlockParam(id, tpe, capt) => BlockParam(id, transform(tpe), capt) } - def transform(definition: Definition)(using PContext): List[Definition] = definition match { - case Definition.Def(id, block) => List(Definition.Def(id, transform(block))) - case Definition.Let(id, tpe, binding) => - val coerce = coercer(binding.tpe, transform(tpe)) - if (coerce.isIdentity) { - List(Definition.Let(id, transform(tpe), transform(binding))) - } else { - val orig = TmpValue("coe") - val origTpe = binding.tpe - List( - Definition.Let(orig, origTpe, transform(binding)), - Definition.Let(id, transform(tpe), coerce(ValueVar(orig, origTpe)))) - } + def transform(toplevel: Toplevel)(using PContext): Toplevel = toplevel match { + case Toplevel.Def(id, block) => Toplevel.Def(id, transform(block)) + case Toplevel.Val(id, tpe, binding) => Toplevel.Val(id, transform(tpe), coerce(transform(binding), transform(tpe))) } def transform(block: Block.BlockLit)(using PContext): Block.BlockLit = block match { @@ -220,6 +213,10 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { case Block.New(impl) => Block.New(transform(impl)) } + def transform(blockVar: BlockVar)(using PContext): BlockVar = blockVar match { + case Block.BlockVar(id, annotatedTpe, annotatedCapt) => + Block.BlockVar(id, transform(annotatedTpe), annotatedCapt) + } def transform(implementation: Implementation)(using PContext): Implementation = implementation match { case Implementation(BlockType.Interface(symbol, targs), operations) => @@ -234,25 +231,22 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { val blockTpe = BlockType.Function(tparams, propTpe.cparams, propTpe.vparams.map(transform), propTpe.bparams.map(transform), transform(propTpe.result)) val implBlock: Block.BlockLit = Block.BlockLit(tparams, cparams, vparams, bparams, transform(body)) - val transformed: Block.BlockLit = coercer(implBlock.tpe, blockTpe)(implBlock) + val transformed: Block.BlockLit = coerce(implBlock, blockTpe) Operation(name, transformed.tparams, transformed.cparams, transformed.vparams, transformed.bparams, transformed.body) } def transform(stmt: Stmt)(using PContext): Stmt = stmt match { - case Stmt.Scope(definitions, body) => - Stmt.Scope(definitions flatMap transform, transform(body)) - case Stmt.Return(expr) => Stmt.Return(transform(expr)) - case Stmt.Val(id, tpe, binding, body) => - val coerce = coercer(binding.tpe, transform(tpe)) - if (coerce.isIdentity) { - Stmt.Val(id, transform(tpe), transform(binding), transform(body)) - } else { - val orig = TmpValue("coe") - Stmt.Val(orig, binding.tpe, transform(binding), - Let(id, transform(binding.tpe), coerce(Pure.ValueVar(orig, binding.tpe)), - transform(body))) + case Stmt.Def(id, block, rest) => + Stmt.Def(id, transform(block), transform(rest)) + case Stmt.Let(id, tpe, binding, rest) => + transform(binding).flatMap { e => coerce(e, transform(tpe)) }.run { e => + Stmt.Let(id, transform(tpe), e, transform(rest)) } + case Stmt.Return(expr) => + Stmt.Return(transform(expr)) + case Stmt.Val(id, tpe, binding, body) => + Stmt.Val(id, transform(tpe), coerce(transform(binding), transform(tpe)), transform(body)) case Stmt.App(callee, targs, vargs, bargs) => val calleeT = transform(callee) val tpe: BlockType.Function = calleeT.tpe match { @@ -260,12 +254,11 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { case _ => sys error "Callee does not have function type" } val itpe = Type.instantiate(tpe, targs, tpe.cparams.map(Set(_))) - val tVargs = vargs map transform - val tBargs = bargs map transform - val vcoercers = (tVargs zip itpe.vparams).map { (a, p) => coercer(a.tpe, p) } - val bcoercers = (tBargs zip itpe.bparams).map { (a, p) => coercer[Block](a.tpe, p) } - val fcoercer = coercer[Block](tpe, itpe, targs) - fcoercer.call(calleeT, (vcoercers zip tVargs).map(_(_)), (bcoercers zip tBargs).map(_(_))) + + val vCoerced = (vargs zip tpe.vparams).map { (v, tpe) => coerce(transform(v), tpe) } // coerce(coerce(transform(v), itpe), itpe, tpe) } + val bCoerced = (bargs zip tpe.bparams).map { (b, tpe) => coerce(transform(b), tpe) } // coerce(coerce(transform(b), itpe), itpe, tpe) } + + coerce(App(calleeT, targs.map(transformArg), vCoerced, bCoerced), itpe.result) // [S](S) => (Int, S) case Stmt.Invoke(callee, method, methodTpe: BlockType.Function, targs, vargs, bargs) => @@ -296,24 +289,12 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { // duplicated from App val itpe = Type.instantiate(methodTpe, targs, methodTpe.cparams.map(Set(_))) - val tVargs = vargs map transform - val tBargs = bargs map transform - val vcoercers = (tVargs zip boxedTpe.vparams).map { (a, p) => coercer(a.tpe, p) } - val bcoercers = (tBargs zip boxedTpe.bparams).map { (a, p) => coercer[Block](a.tpe, p) } + val vCoerced = (vargs zip boxedTpe.vparams).map { (a, tpe) => coerce(transform(a), tpe) } + val bCoerced = (bargs zip boxedTpe.bparams).map { (a, tpe) => coerce(transform(a), tpe) } + // (T, S) (Int, Double) - val rcoercer = coercer(tpe.result, itpe.result) - - val result = Invoke(calleeT, method, boxedTpe, targs.map(transformArg), (vcoercers zip tVargs).map(_(_)), (bcoercers zip tBargs).map(_(_))) - - // (BoxedInt, BoxedDouble) - val out = result.tpe - if (rcoercer.isIdentity) { - result - } else { - val orig = TmpValue("result") - Stmt.Val(orig, out, result, - Stmt.Return(rcoercer(Pure.ValueVar(orig, out)))) - } + coerce(Invoke(calleeT, method, boxedTpe, targs.map(transformArg), vCoerced, bCoerced), itpe.result) + case Stmt.Invoke(callee, method, methodTpe, targs, vargs, bargs) => ??? case Stmt.Get(id, capt, tpe) => Stmt.Get(id, capt, transform(tpe)) @@ -330,7 +311,7 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { val casetpe: BlockType.Function = BlockType.Function(tparams, List(), constructor.fields.map(_.tpe), List(), Type.inferType(clause.body) ) - (id, coercer(clause.tpe, Type.instantiate(casetpe, targs map transformArg, List()))(transform(clause))) + (id, coerce(transform(clause), Type.instantiate(casetpe, targs map transformArg, List()))) }, default map transform) case t => Context.abort(pp"Match on value of type ${t}") } @@ -347,45 +328,19 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { case core.Type.TResume(result, answer) => result case _ => ??? } - val transformedBody = transform(body) - val got = transformedBody.tpe - val doBoxResult = coercer(got, expected) - - if (doBoxResult.isIdentity) { - Stmt.Resume(k, transformedBody) - } else { - val orig = TmpValue("resume_result") - Stmt.Resume(k, - Stmt.Val(orig, got, transformedBody, - Stmt.Return(doBoxResult(Pure.ValueVar(orig, got))))) - } + Stmt.Resume(k, coerce(transform(body), expected)) case Stmt.Region(body) => - val tBody = transform(body) - // make sure the result type is a boxed one - val (expectedBodyTpe, actualReturnType, expectedReturnType) = tBody.tpe match { - case BlockType.Function(tparams, cparams, vparams, bparams, result) => - val boxedResult = transformArg(result) - (BlockType.Function(tparams, cparams, vparams, bparams, boxedResult), boxedResult, result) - case _ => Context.abort("Body of a region cannot have interface type") - } - val doBoxResult = coercer[BlockLit](tBody.tpe, expectedBodyTpe) - // Create coercer for eagerly unboxing the result again - val doUnboxResult = coercer(actualReturnType, expectedReturnType) - val resName = TmpValue("boxedResult") - - if (doUnboxResult.isIdentity && doBoxResult.isIdentity) { - Stmt.Region(tBody) - } else { - Stmt.Val(resName, actualReturnType, Stmt.Region(doBoxResult(tBody)), - Stmt.Return(doUnboxResult(Pure.ValueVar(resName, actualReturnType)))) + transform(body) match { + case BlockLit(tparams, cparams, vparams, bparams, body) => + val originalResult = body.tpe + val boxedResult = transformArg(originalResult) + coerce(Stmt.Region(BlockLit(tparams, cparams, vparams, bparams, coerce(body, boxedResult))), originalResult) } case Stmt.Hole() => Stmt.Hole() } - - - def transform(expr: Expr)(using PContext): Expr = expr match { + def transform(expr: Expr)(using PContext): Bind[Expr] = expr match { case DirectApp(b, targs, vargs, bargs) => val callee = transform(b) val tpe: BlockType.Function = callee.tpe match { @@ -393,14 +348,12 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { case _ => sys error "Callee does not have function type" } val itpe = Type.instantiate(tpe, targs, tpe.cparams.map(Set(_))) - val tVargs = vargs map transform - val tBargs = bargs map transform - val vcoercers = (tVargs zip itpe.vparams).map { (a, p) => coercer(a.tpe, p) } - val bcoercers = (tBargs zip itpe.bparams).map { (a, p) => coercer[Block](a.tpe, p) } - val fcoercer = coercer[Block](tpe, itpe, targs) - fcoercer.callDirect(callee, (vcoercers zip tVargs).map(_(_)), (bcoercers zip tBargs).map(_(_))) - case Run(s) => Run(transform(s)) - case pure: Pure => transform(pure) + val vCoerced = (vargs zip tpe.vparams).map { case (a, tpe) => coerce(transform(a), tpe) } // this was "a.tpe -> itpe -> tpe" + val bCoerced = (bargs zip tpe.bparams).map { case (a, tpe) => coerce(transform(a), tpe) } + + coerce(DirectApp(callee, targs.map(transformArg), vCoerced, bCoerced), itpe.result) + + case pure: Pure => Bind.pure(transform(pure)) } def transform(pure: Pure)(using PContext): Pure = pure match { @@ -413,22 +366,17 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { case _ => sys error "Callee does not have function type" } val itpe = Type.instantiate(tpe, targs, tpe.cparams.map(Set(_))) - val tVargs = vargs map transform - val vcoercers = (tVargs zip itpe.vparams).map { (a, p) => coercer(a.tpe, p) } - val fcoercer = coercer[Block](tpe, itpe, targs) - fcoercer.callPure(b, (vcoercers zip tVargs).map(_(_))) + val vCoerced = (vargs zip tpe.vparams).map { (a, tpe) => coerce(transform(a), tpe) } + coerce(PureApp(callee, targs.map(transformArg), vCoerced), itpe.result) + case Pure.Make(data, tag, vargs) => val dataDecl = PContext.getData(data.name) val ctorDecl = dataDecl.constructors.find(_.id == tag).getOrElse { Context.panic(pp"No constructor found for tag ${tag} in data type: ${data}") } - - val argTypes = vargs.map(_.tpe) val paramTypes = ctorDecl.fields.map(_.tpe) - val coercedArgs = (paramTypes zip (argTypes zip vargs)).map { case (param, (targ, arg)) => - coercer(targ, param)(transform(arg)) - } + val coercedArgs = (vargs zip paramTypes).map { case (arg, paramTpe) => coerce(transform(arg), paramTpe) } Pure.Make(transform(data), tag, coercedArgs) case Pure.Select(target, field, annotatedType) => { @@ -441,7 +389,7 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { val f = fields.find(_.id == field).getOrElse{ Context.abort(s"${id} has no field ${field}.") } - coercer(f.tpe, Type.substitute(f.tpe, (tparams zip targs).toMap, Map()))(Pure.Select(target, field, transform(annotatedType))) + coerce(Pure.Select(target, field, transform(annotatedType)), Type.substitute(f.tpe, (tparams zip targs).toMap, Map())) case t => Context.abort(s"Select on data type ${t.id} is not supported.") } } @@ -478,55 +426,86 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { case ValueType.Boxed(tpe, capt) => ValueType.Boxed(transform(tpe), capt) } - def instantiate[B >: BlockLit <: Block](block: B, targs: List[ValueType])(using PContext): FunctionCoercer[BlockType, B] = { - block.tpe match { - case tpe: BlockType.Function => - coercer(tpe, Type.instantiate(tpe, targs, tpe.cparams.map(Set(_))), targs) - case tpe: BlockType.Interface => - Context.abort(s"Using interface of type ${tpe} in function position.") + + // Coercions + // --------- + def coerce(stmt: Stmt, to: ValueType)(using PContext): Stmt = + val from = stmt.tpe + val coerce = ValueCoercer(from, to) + if (coerce.isIdentity) { stmt } + else { + val orig = TmpValue("coe") + Stmt.Val(orig, coerce.from, stmt, Stmt.Return(coerce(ValueVar(orig, coerce.from)))) } - } - trait Coercer[Ty <: Type, Te <: Tree] { - def from: Ty - def to: Ty + def coerce(expr: Expr, to: ValueType)(using PContext): Bind[Expr] = + val from = expr.tpe + val coerce = ValueCoercer(from, to) + if (coerce.isIdentity) { Bind.pure(expr) } + else { Bind.bind(expr).map { x => coerce(x) } } + + def coerce(pure: Pure, to: ValueType)(using PContext): Pure = ValueCoercer(pure.tpe, to)(pure) + + def coerce(block: Block, to: BlockType)(using PContext): Block = BlockCoercer(block.tpe, to)(block) + + def coerce(block: BlockLit, to: BlockType)(using PContext): BlockLit = BlockCoercer(block.tpe, to)(block) + - @targetName("applyType") - def apply(tpe: Ty): Ty = if tpe == from then to else tpe - def apply(t: Te): Te + sealed trait ValueCoercer { + def from: ValueType + def to: ValueType + def apply(t: Pure): Pure def isIdentity: Boolean = false } - class IdentityCoercer[Ty <: Type, Te <: Tree]( - override val from: Ty, - override val to: Ty) extends Coercer[Ty, Te] { - override def apply(t: Te): Te = t - override def isIdentity: Boolean = true - } - case class BoxCoercer(valueType: ValueType)(using PContext) extends Coercer[ValueType, Pure] { - override def from = valueType - override def to = box(valueType).tpe + object ValueCoercer { + + def apply(from: ValueType, to: ValueType)(using PContext): ValueCoercer = (from, to) match { + case (f, t) if f == t => IdentityCoercer(f, t) + case (_: ValueType.Var, _: ValueType.Var) => IdentityCoercer(from, to) // are always boxed + case (unboxed, boxed) if box.isDefinedAt(unboxed) && box(unboxed).tpe == boxed => BoxCoercer(unboxed) + case (unboxed, _: ValueType.Var) if box.isDefinedAt(unboxed) => BoxCoercer(unboxed) + case (boxed, unboxed) if box.isDefinedAt(unboxed) && box(unboxed).tpe == boxed => UnboxCoercer(unboxed) + case (_: ValueType.Var, unboxed) if box.isDefinedAt(unboxed) => UnboxCoercer(unboxed) + case (unboxed, core.Type.TTop) if box.isDefinedAt(unboxed) => BoxCoercer(unboxed) + case (core.Type.TBottom, unboxed) if box.isDefinedAt(unboxed) => BottomCoercer(unboxed) - override def apply(t: Pure): Pure = { - val boxer = box(valueType) - boxer.box(t) + // assert(cs1 == cs2) // FIXME this seems to fail, what would be the correct check for subcapturing (or similar) here? + case (f @ core.ValueType.Boxed(bt1, cs1), t @ core.ValueType.Boxed(bt2, cs2)) => + new ValueCoercer { + val from: ValueType = f + val to: ValueType = t + private val bcoercer = BlockCoercer(bt1, bt2) + override def isIdentity: Boolean = bcoercer.isIdentity + override def apply(t: Pure): Pure = if isIdentity then t else t match { + case Pure.Box(b, annotatedCapture) => Pure.Box(bcoercer(b), annotatedCapture) + case other => Pure.Box(bcoercer(Block.Unbox(t)), cs2) + } + } + case _ => + //Context.warning(s"Coercing ${PrettyPrinter.format(from)} to ${PrettyPrinter.format(to)}") + IdentityCoercer(from, to) } - } - case class UnboxCoercer(valueType: ValueType)(using PContext) extends Coercer[ValueType, Pure] { - override def from = box(valueType).tpe - override def to = valueType - override def apply(t: Pure): Pure = { - val boxer = box(valueType) - boxer.unbox(t) + class IdentityCoercer(val from: ValueType, val to: ValueType) extends ValueCoercer { + override def apply(t: Pure): Pure = t + override def isIdentity: Boolean = true } - } - case class BottomCoercer(valueType: ValueType)(using PContext) extends Coercer[ValueType, Pure] { - override def from = core.Type.TBottom - override def to = valueType + case class BoxCoercer(tpe: ValueType)(using PContext) extends ValueCoercer { + override def from = tpe + override def to = box(tpe).tpe + override def apply(t: Pure): Pure = box(tpe).box(t) + } + case class UnboxCoercer(tpe: ValueType)(using PContext) extends ValueCoercer { + override def from = box(tpe).tpe + override def to = tpe + override def apply(t: Pure): Pure = box(tpe).unbox(t) + } + case class BottomCoercer(tpe: ValueType)(using PContext) extends ValueCoercer { + override def from = core.Type.TBottom + override def to = tpe - override def apply(t: Pure): Pure = { - to match { - case core.Type.TInt => Pure.Literal(1337L, core.Type.TInt) + override def apply(t: Pure): Pure = to match { + case core.Type.TInt => Pure.Literal(1337L, core.Type.TInt) case core.Type.TDouble => Pure.Literal(13.37, core.Type.TDouble) // Do strings need to be boxed? Really? case core.Type.TString => Pure.Literal("", core.Type.TString) @@ -537,110 +516,58 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { } } - def coercer(from: ValueType, to: ValueType)(using PContext): Coercer[ValueType, Pure] = (from, to) match { - case (f,t) if f==t => new IdentityCoercer(f,t) - case (_: ValueType.Var, _: ValueType.Var) => new IdentityCoercer(from, to) // are always boxed - case (unboxed, boxed) if box.isDefinedAt(unboxed) && box(unboxed).tpe == boxed => BoxCoercer(unboxed) - case (unboxed, _: ValueType.Var) if box.isDefinedAt(unboxed) => BoxCoercer(unboxed) - case (boxed, unboxed) if box.isDefinedAt(unboxed) && box(unboxed).tpe == boxed => UnboxCoercer(unboxed) - case (_: ValueType.Var, unboxed) if box.isDefinedAt(unboxed) => UnboxCoercer(unboxed) - case (unboxed, core.Type.TTop) if box.isDefinedAt(unboxed) => BoxCoercer(unboxed) - case (core.Type.TBottom, unboxed) if box.isDefinedAt(unboxed) => BottomCoercer(unboxed) - case (core.ValueType.Boxed(bt1,cs1), core.ValueType.Boxed(bt2, cs2)) => - // assert(cs1 == cs2) // FIXME this seems to fail, what would be the correct check for subcapturing (or similar) here? - val bcoercer = coercer[Block](bt1, bt2) - if (bcoercer.isIdentity) then { IdentityCoercer(from, to) } else { - val _fr = from - val _to = to - new Coercer[ValueType, Pure] { - val from: ValueType = _fr - val to: ValueType = _to - override def isIdentity: Boolean = false - override def apply(t: Pure): Pure = { - Pure.Box(bcoercer(Block.Unbox(t)), cs2) - } - } - } - case _ => - //Context.warning(s"Coercing ${PrettyPrinter.format(from)} to ${PrettyPrinter.format(to)}") - new IdentityCoercer(from, to) - } + sealed trait BlockCoercer { + def from: BlockType + def to: BlockType - trait FunctionCoercer[Ty <: BlockType, Te <: Block] extends Coercer[Ty, Te] { - def callPure(block: Te, vargs: List[Pure])(using PContext): Pure - def callDirect(block: Te, vargs: List[Pure], bargs: List[Block])(using PContext): Expr - def call(block: Te, vargs: List[Pure], bargs: List[Block])(using PContext): Stmt + def apply[Te >: Block.BlockLit <: Block](t: Te): Te + def isIdentity: Boolean } - class FunctionIdentityCoercer[Ty <: BlockType, Te <: Block]( - from: Ty, to: Ty, targs: List[ValueType]) extends IdentityCoercer[Ty, Te](from, to) with FunctionCoercer[Ty, Te] { - override def call(block: Te, vargs: List[Pure], bargs: List[Block])(using PContext): Stmt = - Stmt.App(block, targs map transformArg, vargs, bargs) - override def callPure(block: Te, vargs: List[Pure])(using PContext): Pure = - Pure.PureApp(block, targs map transformArg, vargs) - override def callDirect(block: Te, vargs: List[Pure], bargs: List[Block])(using PContext): Expr = - DirectApp(block, targs map transformArg, vargs, bargs) - } - def coercer[B >: Block.BlockLit <: Block](fromtpe: BlockType, totpe: BlockType, targs: List[ValueType] = List())(using PContext): FunctionCoercer[BlockType, B] = - (fromtpe, totpe) match { - case (f,t) if f == t => new FunctionIdentityCoercer(fromtpe, totpe, targs) - case (BlockType.Function(ftparams, fcparams, fvparams, fbparams, fresult), - BlockType.Function(ttparams, tcparams, tvparams, tbparams, tresult)) => - - val vcoercers = (fvparams zip tvparams).map { - case (t,f) => // note: Order inversed as contravariant in arguments - coercer(f,t) - } - val bcoercers: List[Coercer[BlockType, Block]] = (fbparams zip tbparams).map { - case (t,f) => // note: Order inversed as contravariant in arguments - coercer(f,t) - } - val rcoercer = coercer(fresult, tresult) + object BlockCoercer { - if((rcoercer +: (vcoercers ++ bcoercers)).forall(_.isIdentity)) { - return new FunctionIdentityCoercer(fromtpe, totpe, targs) // nothing to do here + def apply(from: BlockType, to: BlockType, targs: List[ValueType] = Nil)(using PContext): BlockCoercer = + (from, to) match { + case (f, t) if f == t => IdentityCoercer(f, t) + case (f: BlockType.Function, t: BlockType.Function) => FunctionCoercer(f, t, targs) + case (f: BlockType.Interface, t: BlockType.Interface) => IdentityCoercer(f, t) + case _ => Context.abort(pp"Unsupported coercion from ${from} to ${to}") } - new FunctionCoercer[BlockType, B] { - override def from = fromtpe - override def to = totpe - - override def apply(block: B): B = { - val vparams: List[Param.ValueParam] = vcoercers.map { c => Param.ValueParam(TmpValue("coe"), transform(c.from)) } - val bparams: List[Param.BlockParam] = bcoercers.map { c => val id = TmpBlock("coe"); Param.BlockParam(id, transform(c.from), Set(id)) } - val result = TmpValue("coe") - val inner = TmpBlock() - val vargs = (vcoercers zip vparams).map { case (c, p) => c(Pure.ValueVar(p.id, p.tpe)) } - val bargs = (bcoercers zip bparams).map { case (c, p) => c(Block.BlockVar(p.id, p.tpe, Set.empty)) } - Block.BlockLit(ftparams, bparams.map(_.id), vparams, bparams, - Def(inner, block, - Stmt.Val(result, rcoercer.from, Stmt.App(Block.BlockVar(inner, block.tpe, block.capt), (targs map transformArg) ++ (ftparams map core.ValueType.Var.apply), vargs, bargs), - Stmt.Return(rcoercer(Pure.ValueVar(result, rcoercer.from)))))) - } - - override def callPure(block: B, vargs: List[Pure])(using PContext): Pure = { - rcoercer(Pure.PureApp(block, targs map transformArg, (vcoercers zip vargs).map { case (c,v) => c(v) })) - } - - override def callDirect(block: B, vargs: List[Pure], bargs: List[Block])(using PContext): Expr = { - val result = TmpValue("coe") - Run(Let(result, rcoercer.from, DirectApp(block, targs map transformArg, - (vcoercers zip vargs).map {case (c,v) => c(v)}, - (bcoercers zip bargs).map {case (c,b) => c(b)}), - Return(rcoercer(Pure.ValueVar(result, rcoercer.from))))) - } - - override def call(block: B, vargs: List[Pure], bargs: List[Block])(using PContext): Stmt = { - val result = TmpValue("coe") - Stmt.Val(result, rcoercer.from, Stmt.App(block, targs map transformArg, - (vcoercers zip vargs).map { case (c, v) => c(v) }, - (bcoercers zip bargs).map { case (c, b) => c(b) }), - Return(rcoercer(Pure.ValueVar(result, rcoercer.from)))) - } + class IdentityCoercer(val from: BlockType, val to: BlockType) extends BlockCoercer { + override def apply[Te >: Block.BlockLit <: Block](t: Te): Te = t + override def isIdentity: Boolean = true + } + class FunctionCoercer( + val from: BlockType.Function, + val to: BlockType.Function, + targs: List[ValueType] + )(using PContext) extends BlockCoercer { + + private val BlockType.Function(ftparams, fcparams, fvparams, fbparams, fresult) = from + private val BlockType.Function(ttparams, tcparams, tvparams, tbparams, tresult) = to + + val vcoercers = (fvparams zip tvparams).map { case (t, f) => ValueCoercer(f, t) } + val bcoercers = (fbparams zip tbparams).map { case (t, f) => BlockCoercer(f,t) } + val rcoercer = ValueCoercer(fresult, tresult) + + override def isIdentity = (rcoercer :: vcoercers).forall(_.isIdentity) && bcoercers.forall(_.isIdentity) + + override def apply[Te >: Block.BlockLit <: Block](block: Te): Te = if (isIdentity) block else { + val vparams = vcoercers.map { c => ValueParam(TmpValue("coe"), transform(c.from)) } + val bparams = bcoercers.map { c => val id = TmpBlock("coe"); BlockParam(id, transform(c.from), Set(id)) } + + val inner = TmpBlock() + val vargs = (vcoercers zip vparams).map { case (c, p) => c(Pure.ValueVar(p.id, p.tpe)) } + val bargs = (bcoercers zip bparams).map { case (c, p) => c(Block.BlockVar(p.id, p.tpe, Set.empty)) } + Block.BlockLit(ftparams, bparams.map(_.id), vparams, bparams, + Def(inner, block, + coerce(Stmt.App( + Block.BlockVar(inner, block.tpe, block.capt), + (targs map transformArg) ++ (ftparams map core.ValueType.Var.apply), + vargs, + bargs), tresult))) } - case (BlockType.Interface(n1,targs), BlockType.Interface(n2,_)) => - FunctionIdentityCoercer(fromtpe, totpe, targs) - case _ => Context.abort(pp"Unsupported coercion from ${fromtpe} to ${totpe}") + } } - } diff --git a/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala b/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala index b989fb753..5a06a46e5 100644 --- a/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala +++ b/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala @@ -16,7 +16,7 @@ object PrettyPrinter extends ParenPrettyPrinter { def format(t: ModuleDecl): Document = pretty(toDoc(t), 4) - def format(defs: List[Definition]): String = + def format(defs: List[Toplevel]): String = pretty(toDoc(defs), 60).layout def format(s: Stmt): String = @@ -36,7 +36,7 @@ object PrettyPrinter extends ParenPrettyPrinter { val show: PartialFunction[Any, String] = { case m: ModuleDecl => format(m).layout - case d: Definition => format(List(d)) + case d: Toplevel => format(List(d)) case s: Stmt => format(s) case t: ValueType => format(t) case t: BlockType => format(t) @@ -56,7 +56,7 @@ object PrettyPrinter extends ParenPrettyPrinter { toDoc(m.definitions) } - def toDoc(definitions: List[Definition]): Doc = + def toDoc(definitions: List[Toplevel]): Doc = vsep(definitions map toDoc, semi) def toDoc(e: Extern): Doc = e match { @@ -105,7 +105,6 @@ object PrettyPrinter extends ParenPrettyPrinter { case Select(b, field, tpe) => toDoc(b) <> "." <> toDoc(field) case Box(b, capt) => parens("box" <+> toDoc(b)) - case Run(s) => "run" <+> block(toDoc(s)) } def argsToDoc(targs: List[core.ValueType], vargs: List[core.Pure], bargs: List[core.Block]): Doc = @@ -115,7 +114,7 @@ object PrettyPrinter extends ParenPrettyPrinter { val bargsDoc = bargs.map(toDoc) targsDoc <> parens(vargsDoc ++ bargsDoc) - def paramsToDoc(tps: List[symbols.Symbol], vps: List[Param.ValueParam], bps: List[Param.BlockParam]): Doc = { + def paramsToDoc(tps: List[symbols.Symbol], vps: List[ValueParam], bps: List[BlockParam]): Doc = { val tpsDoc = if (tps.isEmpty) emptyDoc else brackets(tps.map(toDoc)) tpsDoc <> parens(hsep(vps map toDoc, comma)) <> hcat(bps map toDoc) } @@ -152,18 +151,27 @@ object PrettyPrinter extends ParenPrettyPrinter { case Property(name, tpe) => toDoc(name) <> ":" <+> toDoc(tpe) } - def toDoc(d: Definition): Doc = d match { - case Definition.Def(id, BlockLit(tps, cps, vps, bps, body)) => + def toDoc(d: Toplevel): Doc = d match { + case Toplevel.Def(id, BlockLit(tps, cps, vps, bps, body)) => "def" <+> toDoc(id) <> paramsToDoc(tps, vps, bps) <+> "=" <+> block(toDoc(body)) - case Definition.Def(id, block) => + case Toplevel.Def(id, block) => "def" <+> toDoc(id) <+> "=" <+> toDoc(block) - case Definition.Let(id, _, binding) => - "let" <+> toDoc(id) <+> "=" <+> toDoc(binding) + case Toplevel.Val(id, _, binding) => + "vet" <+> toDoc(id) <+> "=" <+> toDoc(binding) } def toDoc(s: Stmt): Doc = s match { - case Scope(definitions, rest) => - toDoc(definitions) <> emptyline <> toDoc(rest) + case Def(id, BlockLit(tps, cps, vps, bps, body), rest) => + "def" <+> toDoc(id) <> paramsToDoc(tps, vps, bps) <+> "=" <+> block(toDoc(body)) <> line <> + toDoc(rest) + + case Def(id, block, rest) => + "def" <+> toDoc(id) <+> "=" <+> toDoc(block) <> line <> + toDoc(rest) + + case Let(id, _, binding, rest) => + "let" <+> toDoc(id) <+> "=" <+> toDoc(binding) <> line <> + toDoc(rest) case Return(e) => "return" <+> toDoc(e) diff --git a/effekt/shared/src/main/scala/effekt/core/Recursive.scala b/effekt/shared/src/main/scala/effekt/core/Recursive.scala index 1e197c89a..898d84073 100644 --- a/effekt/shared/src/main/scala/effekt/core/Recursive.scala +++ b/effekt/shared/src/main/scala/effekt/core/Recursive.scala @@ -16,20 +16,10 @@ class Recursive( val defs: mutable.Map[Id, RecursiveFunction], var stack: List[Id] ) { - def process(d: Definition): Unit = + def process(d: Toplevel): Unit = d match { - case Definition.Def(id, block) => - block match { - case b @ BlockLit(tparams, cparams, vparams, bparams, body) => - defs(id) = RecursiveFunction(b) - val before = stack - stack = id :: stack - process(block) - stack = before - case _ => () - } - case Definition.Let(id, _, binding) => - process(binding) + case Toplevel.Def(id, block) => process(id, block) + case Toplevel.Val(id, _, binding) => process(binding) } def process(b: Block): Unit = @@ -40,15 +30,20 @@ class Recursive( case Block.New(impl) => process(impl) } + def process(id: Id, block: Block): Unit = + block match { + case b : BlockLit => + defs(id) = RecursiveFunction(b) + val before = stack + stack = id :: stack + process(block) + stack = before + case _ => () + } + def process(s: Stmt): Unit = s match { - case Stmt.Scope(definitions, body) => - definitions.foreach { - case d: Definition.Def => - process(d) - case d: Definition.Let => - process(d) - } - process(body) + case Stmt.Def(id, block, body) => process(id, block); process(body) + case Stmt.Let(id, tpe, binding, body) => process(binding); process(body) case Stmt.Return(expr) => process(expr) case Stmt.Val(id, tpe, binding, body) => process(binding); process(body) case a @ Stmt.App(callee, targs, vargs, bargs) => @@ -93,7 +88,6 @@ class Recursive( process(b) vargs.foreach(process) bargs.foreach(process) - case Run(s) => process(s) case Pure.ValueVar(id, annotatedType) => () case Pure.Literal(value, annotatedType) => () case Pure.PureApp(b, targs, vargs) => process(b); vargs.foreach(process) diff --git a/effekt/shared/src/main/scala/effekt/core/Renamer.scala b/effekt/shared/src/main/scala/effekt/core/Renamer.scala index 6d90f3548..9f75e4508 100644 --- a/effekt/shared/src/main/scala/effekt/core/Renamer.scala +++ b/effekt/shared/src/main/scala/effekt/core/Renamer.scala @@ -48,20 +48,13 @@ class Renamer(names: Names = Names(Map.empty), prefix: String = "") extends core } override def stmt: PartialFunction[Stmt, Stmt] = { - case core.Scope(definitions, body) => - - def go(rest: List[Definition], defs: List[Definition]): core.Scope = rest match { - case (d : core.Definition.Def) :: rest => - // can be recursive - withBinding(d.id) { go(rest, defs :+ rewrite(d)) } - case core.Definition.Let(id, tpe, binding) :: rest => - // resolve binding in outer scope - val resolvedBinding = rewrite(binding) - withBinding(id) { go(rest, defs :+ core.Definition.Let(rewrite(id), rewrite(tpe), resolvedBinding)) } - case Nil => core.Scope(defs, rewrite(body)) - } + case core.Def(id, block, body) => + // can be recursive + withBinding(id) { core.Def(rewrite(id), rewrite(block), rewrite(body)) } - go(definitions, Nil) + case core.Let(id, tpe, binding, body) => + val resolvedBinding = rewrite(binding) + withBinding(id) { core.Let(rewrite(id), rewrite(tpe), resolvedBinding, rewrite(body)) } case core.Val(id, tpe, binding, body) => val resolvedBinding = rewrite(binding) diff --git a/effekt/shared/src/main/scala/effekt/core/Transformer.scala b/effekt/shared/src/main/scala/effekt/core/Transformer.scala index c32774a1f..8b77feaf7 100644 --- a/effekt/shared/src/main/scala/effekt/core/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/core/Transformer.scala @@ -48,7 +48,7 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { val exports = transform(mod.exports) val toplevelDeclarations = defs.flatMap(d => transformToplevel(d)) - val definitions = toplevelDeclarations.collect { case d: Definition => d } + val definitions = toplevelDeclarations.collect { case d: Toplevel => d } val externals = toplevelDeclarations.collect { case d: Extern => d } val declarations = toplevelDeclarations.collect { case d: Declaration => d } @@ -61,13 +61,13 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { ModuleDecl(path, mod.includes.map { _.path }, preludeDeclarations ++ declarations, externals, definitions, exports) } - def transformToplevel(d: source.Def)(using Context): List[Definition | Declaration | Extern] = d match { + def transformToplevel(d: source.Def)(using Context): List[Toplevel | Declaration | Extern] = d match { case f @ source.FunDef(id, tps, vps, bps, ret, body) => val tparams = tps.map { p => p.symbol } val cparams = bps.map { b => b.symbol.capture } val vparams = vps map transform val bparams = bps map transform - List(Definition.Def(f.symbol, BlockLit(tparams, cparams, vparams, bparams, transform(body)))) + List(Toplevel.Def(f.symbol, BlockLit(tparams, cparams, vparams, bparams, transform(body)))) case d @ source.DataDef(id, _, ctors) => val datatype = d.symbol @@ -83,7 +83,7 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { case Some(tpe) => transform(tpe) case None => transformed.tpe } - List(Definition.Let(v.symbol, transformedTpe, Run(transformed))) + List(Toplevel.Val(v.symbol, transformedTpe, transformed)) case v @ source.ValDef(id, _, binding) => Context.at(d) { Context.abort("Effectful bindings not allowed on the toplevel") } @@ -91,17 +91,10 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { case v @ source.DefDef(id, annot, binding) => val sym = v.symbol val (definition, bindings) = Context.withBindings { - Definition.Def(sym, transformAsBlock(binding)) + Toplevel.Def(sym, transformAsBlock(binding)) } - // convert binding into Definition. - val additionalDefinitions = bindings.toList.map { - case Binding.Let(name, tpe, binding) => - Definition.Let(name, tpe, binding) - case Binding.Def(name, binding) => Definition.Def(name, binding) - case Binding.Val(name, tpe, binding) => Context.at(d) { Context.abort("Effectful bindings not allowed on the toplevel") } - } - additionalDefinitions ++ List(definition) + bindings.map(core.Binding.toToplevel) ++ List(definition) case _: source.VarDef | _: source.RegDef => Context.at(d) { Context.abort("Mutable variable bindings not allowed on the toplevel") } @@ -157,7 +150,7 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { case source.ExprStmt(e, rest) if pureOrIO(e) => val (expr, bs) = Context.withBindings { transformAsExpr(e) } val let = Let(Wildcard(), expr.tpe, expr, transform(rest)) - Context.reifyBindings(let, bs) + Binding(bs, let) // { e; stmt } --> { val _ = e; stmt } case source.ExprStmt(e, rest) => @@ -180,14 +173,6 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { val bparams = bps map transform Def(f.symbol, BlockLit(tparams, cparams, vparams, bparams, transform(body)), transform(rest)) - case v @ source.ValDef(id, tpe, binding) if pureOrIO(binding) => - val transformed = Run(transform(binding)) - val transformedTpe = v.symbol.tpe match { - case Some(tpe) => transform(tpe) - case None => transformed.tpe - } - Let(v.symbol, transformedTpe, transformed, transform(rest)) - case v @ source.ValDef(id, tpe, binding) => val transformed = transform(binding) val transformedTpe = v.symbol.tpe match { @@ -276,8 +261,8 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { // if this block argument expects to be called using PureApp or DirectApp, make sure it is // by wrapping it in a BlockLit val targs = tparams.map(core.ValueType.Var.apply) - val vparams: List[Param.ValueParam] = vparamtps.map { t => Param.ValueParam(TmpValue("valueParam"), transform(t))} - val vargs = vparams.map { case Param.ValueParam(id, tpe) => Pure.ValueVar(id, tpe) } + val vparams = vparamtps.map { t => core.ValueParam(TmpValue("valueParam"), transform(t))} + val vargs = vparams.map { case core.ValueParam(id, tpe) => Pure.ValueVar(id, tpe) } // [[ f ]] = { (x) => f(x) } def etaExpandPure(b: ExternFunction): BlockLit = { @@ -300,9 +285,9 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { // [[ f ]] = { (x){g} => let r = f(x){g}; return r } def etaExpandDirect(f: ExternFunction): BlockLit = { assert(effects.isEmpty) - val bparams: List[Param.BlockParam] = bparamtps.map { t => val id = TmpBlock("etaParam"); Param.BlockParam(id, transform(t), Set(id)) } + val bparams = bparamtps.map { t => val id = TmpBlock("etaParam"); core.BlockParam(id, transform(t), Set(id)) } val bargs = bparams.map { - case Param.BlockParam(id, tpe, capt) => Block.BlockVar(id, tpe, capt) + case core.BlockParam(id, tpe, capt) => Block.BlockVar(id, tpe, capt) } val result = TmpValue("etaBinding") val resultBinding = DirectApp(BlockVar(f), targs, vargs, bargs) @@ -441,11 +426,11 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { val transformedHandlers = handlers.map { case h @ source.Handler(cap, impl) => val id = h.capability.get.symbol - Definition.Def(id, New(transform(impl, Some(promptVar)))) + Binding.Def(id, New(transform(impl, Some(promptVar)))) } val body: BlockLit = BlockLit(Nil, List(promptCapt), Nil, List(promptParam), - Scope(transformedHandlers, transform(prog))) + Binding(transformedHandlers, transform(prog))) Context.bind(Reset(body)) @@ -493,11 +478,7 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { val blockArgs = bargs.map(transformAsBlock) // val captArgs = blockArgs.map(b => b.capt) //transform(Context.inferredCapture(b))) - if (capture.pureOrIO && bargs.forall { pureOrIO }) { - Run(App(Unbox(e), typeArgs, valueArgs, blockArgs)) - } else { - Context.bind(App(Unbox(e), typeArgs, valueArgs, blockArgs)) - } + Context.bind(App(Unbox(e), typeArgs, valueArgs, blockArgs)) case c @ source.Call(fun: source.IdTarget, _, vargs, bargs) => // assumption: typer removed all ambiguous references, so there is exactly one @@ -574,7 +555,7 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { val resumeCapture = Id("resume") val resumeId = Id("k") val resumeTpe = core.Type.TResume(resultTpe, answerTpe) - val resumeParam: core.BlockParam = core.BlockParam(resumeId, resumeTpe, Set(resumeCapture)) + val resumeParam = core.BlockParam(resumeId, resumeTpe, Set(resumeCapture)) val resumeVar: core.BlockVar = core.BlockVar(resumeId, resumeTpe, Set(resumeCapture)) // (2) eta-expand and bind continuation as a function @@ -585,7 +566,7 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { core.Operation(op.definition, tps, Nil, vps, Nil, core.Shift(prompt, core.BlockLit(Nil, List(resumeCapture), Nil, resumeParam :: Nil, - core.Scope(List(core.Definition.Def(resumeSymbol, resumeFun)), + core.Def(resumeSymbol, resumeFun, transform(body))))) // bi-directional @@ -627,7 +608,7 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { core.Operation(op.definition, tps, cps, vps, bparams, core.Shift(prompt, core.BlockLit(Nil, List(resumeCapture), Nil, resumeParam :: Nil, - core.Scope(List(core.Definition.Def(resumeSymbol, resumeFun)), + core.Stmt.Def(resumeSymbol, resumeFun, transform(body))))) case _ => ??? @@ -778,8 +759,6 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { Context.panic("Should have been translated to a method call!") case f: Field => Context.panic("Should have been translated to a select!") - case f: BlockSymbol if pureOrIO(f) && bargs.forall { pureOrIO } => - Run(App(BlockVar(f), targs, vargsT, bargsT)) case f: BlockSymbol => Context.bind(App(BlockVar(f), targs, vargsT, bargsT)) case f: ValueSymbol => @@ -793,7 +772,7 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { def insertBindings(stmt: => Stmt)(using Context): Stmt = { val (body, bindings) = Context.withBindings { stmt } - Context.reifyBindings(body, bindings) + Binding(bindings, body) } // Translation on Types @@ -864,12 +843,6 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { } -private[core] enum Binding { - case Val(name: TmpValue, tpe: core.ValueType, binding: Stmt) - case Let(name: TmpValue, tpe: core.ValueType, binding: Expr) - case Def(name: BlockSymbol, binding: Block) -} - trait TransformerOps extends ContextOps { Context: Context => /** @@ -916,27 +889,12 @@ trait TransformerOps extends ContextOps { Context: Context => BlockVar(name, b.tpe, b.capt) } - private[core] def withBindings[R](block: => R): (R, ListBuffer[Binding]) = Context in { + private[core] def withBindings[R](block: => R): (R, List[Binding]) = Context in { val before = bindings val b = ListBuffer.empty[Binding] bindings = b val result = block bindings = before - (result, b) - } - - /** - * When reifying bindings, insert let bindings and use RUN when statement is pure or IO. - */ - private[core] def reifyBindings(body: Stmt, bindings: ListBuffer[Binding]): Stmt = { - bindings.foldRight(body) { - // optimization: remove unnecessary binds - case (Binding.Val(x, tpe, b), Return(ValueVar(y, _))) if x == y => b - case (Binding.Val(x, tpe, b), body) => Val(x, tpe, b, body) - case (Binding.Let(x, tpe, Run(s)), Return(ValueVar(y, _))) if x == y => s - case (Binding.Let(x, tpe, b: Pure), Return(ValueVar(y, _))) if x == y => Return(b) - case (Binding.Let(x, tpe, b), body) => Let(x, tpe, b, body) - case (Binding.Def(x, b), body) => Def(x, b, body) - } + (result, b.toList) } } diff --git a/effekt/shared/src/main/scala/effekt/core/Tree.scala b/effekt/shared/src/main/scala/effekt/core/Tree.scala index fef09cfee..d413e11bf 100644 --- a/effekt/shared/src/main/scala/effekt/core/Tree.scala +++ b/effekt/shared/src/main/scala/effekt/core/Tree.scala @@ -33,13 +33,8 @@ import scala.annotation.tailrec * β”‚ │─ [[ StringExternBody ]] * β”‚ │─ [[ Unsupported ]] * β”‚ - * │─ [[ Definition ]] - * β”‚ │─ [[ Def ]] - * β”‚ │─ [[ Let ]] - * β”‚ * │─ [[ Expr ]] * β”‚ │─ [[ DirectApp ]] - * β”‚ │─ [[ Run ]] * β”‚ │─ [[ Pure ]] * β”‚ * │─ [[ Block ]] @@ -48,12 +43,9 @@ import scala.annotation.tailrec * β”‚ │─ [[ Unbox ]] * β”‚ │─ [[ New ]] * β”‚ - * │─ [[ Param ]] - * β”‚ │─ [[ ValueParam ]] - * β”‚ │─ [[ BlockParam ]] - * β”‚ * │─ [[ Stmt ]] - * β”‚ │─ [[ Scope ]] + * β”‚ │─ [[ Def ]] + * β”‚ │─ [[ Let ]] * β”‚ │─ [[ Return ]] * β”‚ │─ [[ Val ]] * β”‚ │─ [[ App ]] @@ -113,7 +105,7 @@ case class ModuleDecl( includes: List[String], declarations: List[Declaration], externs: List[Extern], - definitions: List[Definition], + definitions: List[Toplevel], exports: List[Id] ) extends Tree @@ -136,7 +128,7 @@ case class Property(id: Id, tpe: BlockType) extends Tree * FFI external definitions */ enum Extern extends Tree { - case Def(id: Id, tparams: List[Id], cparams: List[Id], vparams: List[Param.ValueParam], bparams: List[Param.BlockParam], ret: ValueType, annotatedCapture: Captures, body: ExternBody) + case Def(id: Id, tparams: List[Id], cparams: List[Id], vparams: List[ValueParam], bparams: List[BlockParam], ret: ValueType, annotatedCapture: Captures, body: ExternBody) case Include(featureFlag: FeatureFlag, contents: String) } sealed trait ExternBody extends Tree @@ -147,37 +139,18 @@ object ExternBody { } } -enum Definition extends Tree { +enum Toplevel { def id: Id case Def(id: Id, block: Block) - case Let(id: Id, tpe: ValueType, binding: Expr) // PURE on the toplevel? - - // TBD - // case Var(id: Symbol, region: Symbol, init: Pure) // TOPLEVEL could only be {global}, or not at all. - - // TDB - // case Mutual(defs: List[Definition.Def]) - val capt: Captures = Type.inferCapt(this) + case Val(id: Id, tpe: ValueType, binding: core.Stmt) } -// Some smart constructors -private def addToScope(definition: Definition, body: Stmt): Stmt = body match { - case Scope(definitions, body) => Scope(definition :: definitions, body) - case other => Scope(List(definition), other) -} - -def Def(id: Id, block: Block, rest: Stmt) = - addToScope(Definition.Def(id, block), rest) - -def Let(id: Id, tpe: ValueType, binding: Expr, rest: Stmt) = - addToScope(Definition.Let(id, tpe, binding), rest) /** * Expressions (with potential IO effects) * * - [[DirectApp]] - * - [[Run]] * - [[Pure]] */ sealed trait Expr extends Tree { @@ -186,11 +159,7 @@ sealed trait Expr extends Tree { } // invariant, block b is {io}. -case class DirectApp(b: Block, targs: List[ValueType], vargs: List[Pure], bargs: List[Block]) extends Expr - -// only inserted by the transformer if stmt is pure / io -case class Run(s: Stmt) extends Expr - +case class DirectApp(b: Block.BlockVar, targs: List[ValueType], vargs: List[Pure], bargs: List[Block]) extends Expr /** * Pure Expressions (no IO effects, or control effects) @@ -216,7 +185,7 @@ enum Pure extends Expr { /** * Pure FFI calls. Invariant, block b is pure. */ - case PureApp(b: Block, targs: List[ValueType], vargs: List[Pure]) + case PureApp(b: Block.BlockVar, targs: List[ValueType], vargs: List[Pure]) /** * Constructor calls @@ -249,7 +218,7 @@ export Pure.* */ enum Block extends Tree { case BlockVar(id: Id, annotatedTpe: BlockType, annotatedCapt: Captures) - case BlockLit(tparams: List[Id], cparams: List[Id], vparams: List[Param.ValueParam], bparams: List[Param.BlockParam], body: Stmt) + case BlockLit(tparams: List[Id], cparams: List[Id], vparams: List[ValueParam], bparams: List[BlockParam], body: Stmt) case Unbox(pure: Pure) case New(impl: Implementation) @@ -258,13 +227,9 @@ enum Block extends Tree { } export Block.* -enum Param extends Tree { - def id: Id +case class ValueParam(id: Id, tpe: ValueType) +case class BlockParam(id: Id, tpe: BlockType, capt: Captures) - case ValueParam(id: Id, tpe: ValueType) - case BlockParam(id: Id, tpe: BlockType, capt: Captures) -} -export Param.* /** * Statements @@ -272,7 +237,8 @@ export Param.* * ----------[[ effekt.core.Stmt ]]---------- * * ─ [[ Stmt ]] - * │─ [[ Scope ]] + * │─ [[ Def ]] + * │─ [[ Let ]] * │─ [[ Return ]] * │─ [[ Val ]] * │─ [[ App ]] @@ -293,7 +259,9 @@ export Param.* */ enum Stmt extends Tree { - case Scope(definitions: List[Definition], body: Stmt) + // Definitions + case Def(id: Id, block: Block, body: Stmt) + case Let(id: Id, annotatedTpe: ValueType, binding: Expr, body: Stmt) // Fine-grain CBV case Return(expr: Pure) @@ -337,19 +305,6 @@ enum Stmt extends Tree { } export Stmt.* -/** - * A smart constructor for `stmt.Scope` that only introduces a scope if there are bindings - */ -def MaybeScope(definitions: List[Definition], body: Stmt): Stmt = body match { - // flatten scopes - // { def f = ...; { def g = ...; BODY } } = { def f = ...; def g; BODY } - case Stmt.Scope(others, body) => MaybeScope(definitions ++ others, body) - - // Drop scope if empty - // { ; BODY } = BODY - case _ if definitions.isEmpty => body - case _ => Stmt.Scope(definitions, body) -} /** * An instance of an interface, concretely implementing the operations. @@ -366,10 +321,64 @@ case class Implementation(interface: BlockType.Interface, operations: List[Opera * * TODO drop resume here since it is not needed anymore... */ -case class Operation(name: Id, tparams: List[Id], cparams: List[Id], vparams: List[Param.ValueParam], bparams: List[Param.BlockParam], body: Stmt) { +case class Operation(name: Id, tparams: List[Id], cparams: List[Id], vparams: List[ValueParam], bparams: List[BlockParam], body: Stmt) { val capt = body.capt -- cparams.toSet } +/** + * Bindings are not part of the tree but used in transformations + */ +private[core] enum Binding { + case Val(id: Id, tpe: ValueType, binding: Stmt) + case Let(id: Id, tpe: ValueType, binding: Expr) + case Def(id: Id, binding: Block) + + def id: Id +} +private[core] object Binding { + def apply(bindings: List[Binding], body: Stmt): Stmt = bindings match { + case Nil => body + case Binding.Val(name, tpe, binding) :: rest => Stmt.Val(name, tpe, binding, Binding(rest, body)) + case Binding.Let(name, tpe, binding) :: rest => Stmt.Let(name, tpe, binding, Binding(rest, body)) + case Binding.Def(name, binding) :: rest => Stmt.Def(name, binding, Binding(rest, body)) + } + + def toToplevel(b: Binding): Toplevel = b match { + case Binding.Val(name, tpe, binding) => Toplevel.Val(name, tpe, binding) + case Binding.Let(name, tpe, binding) => ??? //Toplevel.Val(name, tpe, Stmt.Return(binding)) + case Binding.Def(name, binding) => Toplevel.Def(name, binding) + } +} + +// Binding Monad +// ------------- +case class Bind[+A](value: A, bindings: List[Binding]) { + def run(f: A => Stmt): Stmt = Binding(bindings, f(value)) + def map[B](f: A => B): Bind[B] = Bind(f(value), bindings) + def flatMap[B](f: A => Bind[B]): Bind[B] = + val Bind(result, other) = f(value) + Bind(result, bindings ++ other) + def apply[B](f: A => Bind[B]): Bind[B] = flatMap(f) +} +object Bind { + def pure[A](value: A): Bind[A] = Bind(value, Nil) + def bind[A](expr: Expr): Bind[ValueVar] = + val id = Id("tmp") + Bind(ValueVar(id, expr.tpe), List(Binding.Let(id, expr.tpe, expr))) + + def bind[A](block: Block): Bind[BlockVar] = + val id = Id("tmp") + Bind(BlockVar(id, block.tpe, block.capt), List(Binding.Def(id, block))) + + def delimit(b: Bind[Stmt]): Stmt = b.run(a => a) + + def traverse[S, T](l: List[S])(f: S => Bind[T]): Bind[List[T]] = + l match { + case Nil => pure(Nil) + case head :: tail => for { x <- f(head); xs <- traverse(tail)(f) } yield x :: xs + } +} + object Tree { @@ -397,10 +406,9 @@ object Tree { def expr(using Ctx): PartialFunction[Expr, Res] = PartialFunction.empty def stmt(using Ctx): PartialFunction[Stmt, Res] = PartialFunction.empty def block(using Ctx): PartialFunction[Block, Res] = PartialFunction.empty - def defn(using Ctx): PartialFunction[Definition, Res] = PartialFunction.empty - def impl(using Ctx): PartialFunction[Implementation, Res] = PartialFunction.empty + def toplevel(using Ctx): PartialFunction[Toplevel, Res] = PartialFunction.empty + def implementation(using Ctx): PartialFunction[Implementation, Res] = PartialFunction.empty def operation(using Ctx): PartialFunction[Operation, Res] = PartialFunction.empty - def param(using Ctx): PartialFunction[Param, Res] = PartialFunction.empty def clause(using Ctx): PartialFunction[(Id, BlockLit), Res] = PartialFunction.empty def externBody(using Ctx): PartialFunction[ExternBody, Res] = PartialFunction.empty @@ -417,8 +425,8 @@ object Tree { def query(e: Expr)(using Ctx): Res = structuralQuery(e, expr) def query(s: Stmt)(using Ctx): Res = structuralQuery(s, stmt) def query(b: Block)(using Ctx): Res = structuralQuery(b, block) - def query(d: Definition)(using Ctx): Res = structuralQuery(d, defn) - def query(d: Implementation)(using Ctx): Res = structuralQuery(d, impl) + def query(d: Toplevel)(using Ctx): Res = structuralQuery(d, toplevel) + def query(d: Implementation)(using Ctx): Res = structuralQuery(d, implementation) def query(d: Operation)(using Ctx): Res = structuralQuery(d, operation) def query(matchClause: (Id, BlockLit))(using Ctx): Res = if clause.isDefinedAt(matchClause) then clause.apply(matchClause) else matchClause match { @@ -432,22 +440,20 @@ object Tree { def pure: PartialFunction[Pure, Pure] = PartialFunction.empty def expr: PartialFunction[Expr, Expr] = PartialFunction.empty def stmt: PartialFunction[Stmt, Stmt] = PartialFunction.empty - def defn: PartialFunction[Definition, Definition] = PartialFunction.empty + def toplevel: PartialFunction[Toplevel, Toplevel] = PartialFunction.empty def block: PartialFunction[Block, Block] = PartialFunction.empty - def handler: PartialFunction[Implementation, Implementation] = PartialFunction.empty - def param: PartialFunction[Param, Param] = PartialFunction.empty + def implementation: PartialFunction[Implementation, Implementation] = PartialFunction.empty def rewrite(x: Id): Id = if id.isDefinedAt(x) then id(x) else x def rewrite(p: Pure): Pure = rewriteStructurally(p, pure) def rewrite(e: Expr): Expr = rewriteStructurally(e, expr) def rewrite(s: Stmt): Stmt = rewriteStructurally(s, stmt) def rewrite(b: Block): Block = rewriteStructurally(b, block) - def rewrite(d: Definition): Definition = rewriteStructurally(d, defn) - def rewrite(e: Implementation): Implementation = rewriteStructurally(e, handler) + def rewrite(d: Toplevel): Toplevel = rewriteStructurally(d, toplevel) + def rewrite(e: Implementation): Implementation = rewriteStructurally(e, implementation) def rewrite(o: Operation): Operation = rewriteStructurally(o) - def rewrite(p: Param): Param = rewriteStructurally(p, param) - def rewrite(p: Param.ValueParam): Param.ValueParam = rewrite(p: Param).asInstanceOf[Param.ValueParam] - def rewrite(p: Param.BlockParam): Param.BlockParam = rewrite(p: Param).asInstanceOf[Param.BlockParam] + def rewrite(p: ValueParam): ValueParam = rewriteStructurally(p) + def rewrite(p: BlockParam): BlockParam = rewriteStructurally(p) def rewrite(b: ExternBody): ExternBody= rewrite(b) def rewrite(b: BlockLit): BlockLit = if block.isDefinedAt(b) then block(b).asInstanceOf else b match { @@ -481,22 +487,20 @@ object Tree { def pure(using Ctx): PartialFunction[Pure, Pure] = PartialFunction.empty def expr(using Ctx): PartialFunction[Expr, Expr] = PartialFunction.empty def stmt(using Ctx): PartialFunction[Stmt, Stmt] = PartialFunction.empty - def defn(using Ctx): PartialFunction[Definition, Definition] = PartialFunction.empty + def toplevel(using Ctx): PartialFunction[Toplevel, Toplevel] = PartialFunction.empty def block(using Ctx): PartialFunction[Block, Block] = PartialFunction.empty - def handler(using Ctx): PartialFunction[Implementation, Implementation] = PartialFunction.empty - def param(using Ctx): PartialFunction[Param, Param] = PartialFunction.empty + def implementation(using Ctx): PartialFunction[Implementation, Implementation] = PartialFunction.empty def rewrite(x: Id)(using Ctx): Id = if id.isDefinedAt(x) then id(x) else x def rewrite(p: Pure)(using Ctx): Pure = rewriteStructurally(p, pure) def rewrite(e: Expr)(using Ctx): Expr = rewriteStructurally(e, expr) def rewrite(s: Stmt)(using Ctx): Stmt = rewriteStructurally(s, stmt) def rewrite(b: Block)(using Ctx): Block = rewriteStructurally(b, block) - def rewrite(d: Definition)(using Ctx): Definition = rewriteStructurally(d, defn) - def rewrite(e: Implementation)(using Ctx): Implementation = rewriteStructurally(e, handler) + def rewrite(d: Toplevel)(using Ctx): Toplevel = rewriteStructurally(d, toplevel) + def rewrite(e: Implementation)(using Ctx): Implementation = rewriteStructurally(e, implementation) def rewrite(o: Operation)(using Ctx): Operation = rewriteStructurally(o) - def rewrite(p: Param)(using Ctx): Param = rewriteStructurally(p, param) - def rewrite(p: Param.ValueParam)(using Ctx): Param.ValueParam = rewrite(p: Param).asInstanceOf[Param.ValueParam] - def rewrite(p: Param.BlockParam)(using Ctx): Param.BlockParam = rewrite(p: Param).asInstanceOf[Param.BlockParam] + def rewrite(p: ValueParam)(using Ctx): ValueParam = rewriteStructurally(p) + def rewrite(p: BlockParam)(using Ctx): BlockParam = rewriteStructurally(p) def rewrite(b: ExternBody)(using Ctx): ExternBody= rewrite(b) def rewrite(b: BlockLit)(using Ctx): BlockLit = if block.isDefinedAt(b) then block(b).asInstanceOf else b match { @@ -573,7 +577,6 @@ object Variables { def free(e: Expr): Variables = e match { case DirectApp(b, targs, vargs, bargs) => free(b) ++ all(vargs, free) ++ all(bargs, free) - case Run(s) => free(s) case Pure.ValueVar(id, annotatedType) => Variables.value(id, annotatedType) case Pure.Literal(value, annotatedType) => Variables.empty case Pure.PureApp(b, targs, vargs) => free(b) ++ all(vargs, free) @@ -590,9 +593,9 @@ object Variables { case Block.New(impl) => free(impl) } - def free(d: Definition): Variables = d match { - case Definition.Def(id, block) => free(block) - id - case Definition.Let(id, _, binding) => free(binding) + def free(d: Toplevel): Variables = d match { + case Toplevel.Def(id, block) => free(block) - id + case Toplevel.Val(id, _, binding) => free(binding) } def all[T](t: IterableOnce[T], f: T => Variables): Variables = @@ -605,16 +608,8 @@ object Variables { free(body) -- all(vparams, bound) -- all(bparams, bound) } def free(s: Stmt): Variables = s match { - // currently local functions cannot be mutually recursive - case Stmt.Scope(defs, body) => - var stillFree = Variables.empty - var boundSoFar = Variables.empty - defs.foreach { d => - stillFree = stillFree ++ (free(d) -- boundSoFar -- bound(d)) - boundSoFar = boundSoFar ++ bound(d) - } - stillFree ++ (free(body) -- boundSoFar) - + case Stmt.Def(id, block, body) => (free(block) ++ free(body)) -- Variables.block(id, block.tpe, block.capt) + case Stmt.Let(id, tpe, binding, body) => free(binding) ++ (free(body) -- Variables.value(id, binding.tpe)) case Stmt.Return(expr) => free(expr) case Stmt.Val(id, tpe, binding, body) => free(binding) ++ (free(body) -- Variables.value(id, binding.tpe)) case Stmt.App(callee, targs, vargs, bargs) => free(callee) ++ all(vargs, free) ++ all(bargs, free) @@ -640,10 +635,9 @@ object Variables { def bound(t: ValueParam): Variables = Variables.value(t.id, t.tpe) def bound(t: BlockParam): Variables = Variables.block(t.id, t.tpe, t.capt) - - def bound(d: Definition): Variables = d match { - case Definition.Def(id, block) => Variables.block(id, block.tpe, block.capt) - case Definition.Let(id, tpe, binding) => Variables.value(id, tpe) + def bound(t: Toplevel): Variables = t match { + case Toplevel.Def(id, block) => Variables.block(id, block.tpe, block.capt) + case Toplevel.Val(id, tpe, binding) => Variables.value(id, tpe) } } @@ -661,15 +655,8 @@ object substitutions { def shadowValues(shadowed: IterableOnce[Id]): Substitution = copy(values = values -- shadowed) def shadowBlocks(shadowed: IterableOnce[Id]): Substitution = copy(blocks = blocks -- shadowed) - def shadowDefinitions(shadowed: Seq[Definition]): Substitution = copy( - values = values -- shadowed.collect { case d: Definition.Let => d.id }, - blocks = blocks -- shadowed.collect { case d: Definition.Def => d.id } - ) - - def shadowParams(shadowed: Seq[Param]): Substitution = copy( - values = values -- shadowed.collect { case d: Param.ValueParam => d.id }, - blocks = blocks -- shadowed.collect { case d: Param.BlockParam => d.id } - ) + def shadowParams(vparams: Seq[ValueParam], bparams: Seq[BlockParam]): Substitution = + copy(values = values -- vparams.map(_.id), blocks = blocks -- bparams.map(_.id)) } // Starting point for inlining, creates Maps(params -> args) and passes to normal substitute @@ -684,20 +671,12 @@ object substitutions { substitute(body)(using Substitution(tSubst, cSubst, vSubst, bSubst)) } - //Replaces all variables contained in one of the Maps with their value - def substitute(definition: Definition)(using Substitution): Definition = - definition match { - case Definition.Def(id, block) => Definition.Def(id, substitute(block)) - case Definition.Let(id, tpe, binding) => Definition.Let(id, tpe, substitute(binding)) - } - def substitute(expression: Expr)(using Substitution): Expr = expression match { - case DirectApp(b, targs, vargs, bargs) => - DirectApp(substitute(b), targs.map(substitute), vargs.map(substitute), bargs.map(substitute)) - - case Run(s) => - Run(substitute(s)) + case DirectApp(f, targs, vargs, bargs) => substitute(f) match { + case g : Block.BlockVar => DirectApp(g, targs.map(substitute), vargs.map(substitute), bargs.map(substitute)) + case _ => INTERNAL_ERROR("Should never substitute a concrete block for an FFI function.") + } case p: Pure => substitute(p) @@ -705,9 +684,13 @@ object substitutions { def substitute(statement: Stmt)(using subst: Substitution): Stmt = statement match { - case Scope(definitions, body) => - Scope(definitions.map(substitute), - substitute(body)(using subst shadowDefinitions definitions)) + case Def(id, block, body) => + Def(id, substitute(block)(using subst shadowBlocks List(id)), + substitute(body)(using subst shadowBlocks List(id))) + + case Let(id, tpe, binding, body) => + Let(id, substitute(tpe), substitute(binding), + substitute(body)(using subst shadowValues List(id))) case Return(expr) => Return(substitute(expr)) @@ -765,7 +748,7 @@ object substitutions { BlockLit(tparams, cparams, vparams.map(p => substitute(p)(using shadowedTypelevel)), bparams.map(p => substitute(p)(using shadowedTypelevel)), - substitute(body)(using shadowedTypelevel shadowParams (vparams ++ bparams))) + substitute(body)(using shadowedTypelevel.shadowParams(vparams, bparams))) } def substituteAsVar(id: Id)(using subst: Substitution): Id = @@ -794,8 +777,10 @@ object substitutions { case Make(tpe, tag, vargs) => Make(substitute(tpe).asInstanceOf, tag, vargs.map(substitute)) - case PureApp(b, targs, vargs) => - PureApp(substitute(b), targs.map(substitute), vargs.map(substitute)) + case PureApp(f, targs, vargs) => substitute(f) match { + case g : Block.BlockVar => PureApp(g, targs.map(substitute), vargs.map(substitute)) + case _ => INTERNAL_ERROR("Should never substitute a concrete block for an FFI function.") + } case Select(target, field, annotatedType) => Select(substitute(target), field, substitute(annotatedType)) @@ -817,17 +802,17 @@ object substitutions { Operation(name, tparams, cparams, vparams.map(p => substitute(p)(using shadowedTypelevel)), bparams.map(p => substitute(p)(using shadowedTypelevel)), - substitute(body)(using shadowedTypelevel shadowParams (vparams ++ bparams))) + substitute(body)(using shadowedTypelevel.shadowParams(vparams, bparams))) } - def substitute(param: Param.ValueParam)(using Substitution): Param.ValueParam = + def substitute(param: ValueParam)(using Substitution): ValueParam = param match { - case Param.ValueParam(id, tpe) => Param.ValueParam(id, substitute(tpe)) + case ValueParam(id, tpe) => ValueParam(id, substitute(tpe)) } - def substitute(param: Param.BlockParam)(using Substitution): Param.BlockParam = + def substitute(param: BlockParam)(using Substitution): BlockParam = param match { - case Param.BlockParam(id, tpe, capt) => Param.BlockParam(id, substitute(tpe), substitute(capt)) + case BlockParam(id, tpe, capt) => BlockParam(id, substitute(tpe), substitute(capt)) } def substitute(tpe: ValueType)(using subst: Substitution): ValueType = diff --git a/effekt/shared/src/main/scala/effekt/core/Type.scala b/effekt/shared/src/main/scala/effekt/core/Type.scala index 2b94fc0ea..c03d72330 100644 --- a/effekt/shared/src/main/scala/effekt/core/Type.scala +++ b/effekt/shared/src/main/scala/effekt/core/Type.scala @@ -189,7 +189,8 @@ object Type { } def inferType(stmt: Stmt): ValueType = stmt match { - case Stmt.Scope(definitions, body) => body.tpe + case Stmt.Def(id, block, body) => body.tpe + case Stmt.Let(id, tpe, binding, body) => body.tpe case Stmt.Return(expr) => expr.tpe case Stmt.Val(id, tpe, binding, body) => body.tpe case Stmt.App(callee, targs, vargs, bargs) => @@ -219,13 +220,9 @@ object Type { case Stmt.Hole() => TBottom } - def inferCapt(defn: Definition): Captures = defn match { - case Definition.Def(id, block) => block.capt - case Definition.Let(id, tpe, binding) => binding.capt - } - def inferCapt(stmt: Stmt): Captures = stmt match { - case Stmt.Scope(definitions, body) => definitions.foldLeft(body.capt)(_ ++ _.capt) + case Stmt.Def(id, block, body) => block.capt ++ body.capt + case Stmt.Let(id, tpe, binding, body) => binding.capt ++ body.capt case Stmt.Return(expr) => Set.empty case Stmt.Val(id, tpe, binding, body) => binding.capt ++ body.capt case Stmt.App(callee, targs, vargs, bargs) => callee.capt ++ bargs.flatMap(_.capt).toSet @@ -246,7 +243,6 @@ object Type { def inferType(expr: Expr): ValueType = expr match { case DirectApp(callee, targs, vargs, bargs) => instantiate(callee.functionType, targs, bargs.map(_.capt)).result - case Run(s) => s.tpe case Pure.ValueVar(id, tpe) => tpe case Pure.Literal(value, tpe) => tpe case Pure.PureApp(callee, targs, args) => instantiate(callee.functionType, targs, Nil).result @@ -261,7 +257,6 @@ object Type { def inferCapt(expr: Expr): Captures = expr match { case DirectApp(callee, targs, vargs, bargs) => callee.capt ++ bargs.flatMap(_.capt).toSet - case Run(s) => s.capt case pure: Pure => Set.empty } diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/BindSubexpressions.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/BindSubexpressions.scala index 52ebfd34e..e4964677d 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/BindSubexpressions.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/BindSubexpressions.scala @@ -2,6 +2,8 @@ package effekt package core package optimizer +import core.Bind.* + // Establishes a normal form in which every subexpression // is explicitly named and aliasing (val x = y) is removed. // @@ -20,39 +22,34 @@ object BindSubexpressions { def transform(m: ModuleDecl): ModuleDecl = m match { case ModuleDecl(path, includes, declarations, externs, definitions, exports) => - val (newDefs, env) = transformDefs(definitions)(using Map.empty) - ModuleDecl(path, includes, declarations, externs, newDefs, exports) + given Env = Map.empty + ModuleDecl(path, includes, declarations, externs, transformToplevel(definitions), exports) } - def transformDefs(definitions: List[Definition])(using env: Env): (List[Definition], Env) = - var definitionsSoFar = List.empty[Definition] - var envSoFar = env - - definitions.foreach { - case Definition.Def(id, block) => - transform(block)(using envSoFar) match { - case Bind(Block.BlockVar(x, _, _), defs) => - definitionsSoFar ++= defs - envSoFar = alias(id, x, envSoFar) - - case Bind(other, defs) => - definitionsSoFar = definitionsSoFar ++ (defs :+ Definition.Def(id, other)) - } - case Definition.Let(id, tpe, expr) => - transform(expr)(using envSoFar) match { - case Bind(Pure.ValueVar(x, _), defs) => - definitionsSoFar ++= defs - envSoFar = alias(id, x, envSoFar) - case Bind(other, defs) => - definitionsSoFar = definitionsSoFar ++ (defs :+ Definition.Let(id, transform(tpe)(using envSoFar), other)) - } + def transformToplevel(definitions: List[Toplevel])(using env: Env): List[Toplevel] = + definitions.flatMap { + case Toplevel.Def(id, block) => transform(block) match { + case Bind(block, bindings) => bindings.map(Binding.toToplevel) :+ Toplevel.Def(id, block) + + } + case Toplevel.Val(id, tpe, binding) => Toplevel.Val(id, transform(tpe), transform(binding)) :: Nil } - (definitionsSoFar, envSoFar) def transform(s: Stmt)(using env: Env): Stmt = s match { - case Stmt.Scope(definitions, body) => - val (newDefs, newEnv) = transformDefs(definitions) - MaybeScope(newDefs, transform(body)(using newEnv)) + + case Stmt.Def(id, block, body) => transform(block) match { + case Bind(Block.BlockVar(x, _, _), bindings) => + Binding(bindings, transform(body)(using alias(id, x, env))) + case Bind(other, bindings) => + Binding(bindings, Stmt.Def(id, other, transform(body))) + } + + case Stmt.Let(id, tpe, binding, body) => transform(binding) match { + case Bind(Pure.ValueVar(x, _), bindings) => + Binding(bindings, transform(body)(using alias(id, x, env))) + case Bind(other, bindings) => + Binding(bindings, Stmt.Let(id, tpe, other, transform(body))) + } case Stmt.App(callee, targs, vargs, bargs) => delimit { for { @@ -132,21 +129,17 @@ object BindSubexpressions { case Pure.Make(data, tag, vargs) => transformExprs(vargs) { vs => bind(Pure.Make(data, tag, vs)) } - case DirectApp(block, targs, vargs, bargs) => for { - b <- transform(block); + case DirectApp(f, targs, vargs, bargs) => for { vs <- transformExprs(vargs); bs <- transformBlocks(bargs); - res <- bind(DirectApp(b, targs.map(transform), vs, bs)) + res <- bind(DirectApp(f, targs.map(transform), vs, bs)) } yield res - case Pure.PureApp(block, targs, vargs) => for { - b <- transform(block); + case Pure.PureApp(f, targs, vargs) => for { vs <- transformExprs(vargs); - res <- bind(Pure.PureApp(b, targs.map(transform), vs)) + res <- bind(Pure.PureApp(f, targs.map(transform), vs)) } yield res case Pure.Select(target, field, tpe) => transform(target) { v => bind(Pure.Select(v, field, transform(tpe))) } case Pure.Box(block, capt) => transform(block) { b => bind(Pure.Box(b, transform(capt))) } - - case Run(s) => bind(Run(transform(s))) } def transformExprs(es: List[Expr])(using Env): Bind[List[ValueVar | Literal]] = traverse(es)(transform) @@ -167,33 +160,4 @@ object BindSubexpressions { BlockType.Interface(name, targs.map(transform)) } def transform(captures: Captures)(using Env): Captures = captures.map(transform) - - - // Binding Monad - // ------------- - case class Bind[+A](value: A, definitions: List[Definition]) { - def run(f: A => Stmt): Stmt = MaybeScope(definitions, f(value)) - def map[B](f: A => B): Bind[B] = Bind(f(value), definitions) - def flatMap[B](f: A => Bind[B]): Bind[B] = - val Bind(result, other) = f(value) - Bind(result, definitions ++ other) - def apply[B](f: A => Bind[B]): Bind[B] = flatMap(f) - } - def pure[A](value: A): Bind[A] = Bind(value, Nil) - def bind[A](expr: Expr): Bind[ValueVar] = - val id = Id("tmp") - Bind(ValueVar(id, expr.tpe), List(Definition.Let(id, expr.tpe, expr))) - - def bind[A](block: Block): Bind[BlockVar] = - val id = Id("tmp") - Bind(BlockVar(id, block.tpe, block.capt), List(Definition.Def(id, block))) - - def delimit(b: Bind[Stmt]): Stmt = b.run(a => a) - - def traverse[S, T](l: List[S])(f: S => Bind[T]): Bind[List[T]] = - l match { - case Nil => pure(Nil) - case head :: tail => for { x <- f(head); xs <- traverse(tail)(f) } yield x :: xs - } - } diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/Deadcode.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/Deadcode.scala index 649e5c28b..a9934a286 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/Deadcode.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/Deadcode.scala @@ -6,14 +6,10 @@ class Deadcode(reachable: Map[Id, Usage]) extends core.Tree.Rewrite { override def stmt = { // Remove local unused definitions - case Scope(defs, stmt) => - MaybeScope(defs.collect { - case d: Definition.Def if reachable.isDefinedAt(d.id) => rewrite(d) - // we only keep non-pure OR reachable let bindings - case d: Definition.Let if d.capt.nonEmpty || reachable.isDefinedAt(d.id) => rewrite(d) - }, rewrite(stmt)) + case Stmt.Def(id, block, body) if !reachable.isDefinedAt(id) => rewrite(body) + case Stmt.Let(id, tpe, binding, body) if binding.capt.isEmpty && !reachable.isDefinedAt(id) => rewrite(body) - case Reset(body) => + case Stmt.Reset(body) => rewrite(body) match { case BlockLit(tparams, cparams, vparams, List(prompt), body) if !reachable.isDefinedAt(prompt.id) => body case b => Stmt.Reset(b) diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/DropBindings.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/DropBindings.scala index 740586c47..7afe05cf0 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/DropBindings.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/DropBindings.scala @@ -50,17 +50,10 @@ object DropBindings extends Phase[CoreTransformed, CoreTransformed] { case Pure.ValueVar(id, tpe) if usedOnce(id) && hasDefinition(id) => definitionOf(id) } - override def stmt(using DropContext) = { - case Stmt.Scope(defs, body) => - var contextSoFar = currentContext - val ds = defs.flatMap { - case Definition.Let(id, tpe, p: Pure) if usedOnce(id) => - val transformed = rewrite(p)(using contextSoFar) - contextSoFar = contextSoFar.updated(id, transformed) - None - case d => Some(rewrite(d)(using contextSoFar)) - } - MaybeScope(ds, rewrite(body)(using contextSoFar)) + override def stmt(using C: DropContext) = { + case Stmt.Let(id, tpe, p: Pure, body) if usedOnce(id) => + val transformed = rewrite(p) + rewrite(body)(using C.updated(id, transformed)) } } } diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala index 2744f255b..e6a548ed0 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala @@ -70,25 +70,32 @@ object Normalizer { normal => val usage = Reachable(entrypoints, m) val defs = m.definitions.collect { - case Definition.Def(id, block) => id -> block + case Toplevel.Def(id, block) => id -> block }.toMap val context = Context(defs, Map.empty, DeclarationContext(m.declarations, m.externs), mutable.Map.from(usage), maxInlineSize) - val (normalizedDefs, _) = normalize(m.definitions)(using context) + val (normalizedDefs, _) = normalizeToplevel(m.definitions)(using context) m.copy(definitions = normalizedDefs) } - def normalize(definitions: List[Definition])(using ctx: Context): (List[Definition], Context) = + def normalizeToplevel(definitions: List[Toplevel])(using ctx: Context): (List[Toplevel], Context) = var contextSoFar = ctx val defs = definitions.map { - case Definition.Def(id, block) => - val normalized = active(block)(using contextSoFar).dealiased + case Toplevel.Def(id, block) => + val normalized = normalize(block)(using contextSoFar) contextSoFar = contextSoFar.bind(id, normalized) - Definition.Def(id, normalized) - case Definition.Let(id, tpe, expr) => - val normalized = active(expr)(using contextSoFar) - contextSoFar = contextSoFar.bind(id, normalized) - Definition.Let(id, tpe, normalized) + Toplevel.Def(id, normalized) + + case Toplevel.Val(id, tpe, binding) => + // TODO commute (similar to normalizeVal) + // val foo = { val bar = ...; ... } = val bar = ...; val foo = ...; + val normalized = normalize(binding)(using contextSoFar) + normalized match { + case Stmt.Return(expr) => + contextSoFar = contextSoFar.bind(id, expr) + case normalized => () + } + Toplevel.Val(id, tpe, normalized) } (defs, contextSoFar) @@ -150,16 +157,15 @@ object Normalizer { normal => case other => other // stuck } - def normalize(d: Definition)(using C: Context): Definition = d match { - case Definition.Def(id, block) => Definition.Def(id, normalize(block)) - case Definition.Let(id, tpe, binding) => Definition.Let(id, tpe, normalize(binding)) - } - def normalize(s: Stmt)(using C: Context): Stmt = s match { - case Stmt.Scope(definitions, body) => - val (defs, ctx) = normalize(definitions) - normal.Scope(defs, normalize(body)(using ctx)) + case Stmt.Def(id, block, body) => + val normalized = active(block).dealiased + Stmt.Def(id, normalized, normalize(body)(using C.bind(id, normalized))) + + case Stmt.Let(id, tpe, expr, body) => + val normalized = active(expr) + Stmt.Let(id, tpe, normalized, normalize(body)(using C.bind(id, normalized))) // Redexes // ------- @@ -206,21 +212,24 @@ object Normalizer { normal => def normalizeVal(id: Id, tpe: ValueType, binding: Stmt, body: Stmt): Stmt = binding match { // [[ val x = return e; s ]] = let x = [[ e ]]; [[ s ]] - case Stmt.Return(expr) => - normal.Scope(List(Definition.Let(id, tpe, expr)), - normalize(body)(using C.bind(id, expr))) + case Stmt.Return(expr2) => + Stmt.Let(id, tpe, expr2, normalize(body)(using C.bind(id, expr2))) + + // Commute val and bindings + // [[ val x = { def f = ...; STMT }; STMT ]] = def f = ...; val x = STMT; STMT + case Stmt.Def(id2, block2, body2) => + Stmt.Def(id2, block2, normalizeVal(id, tpe, body2, body)) // Commute val and bindings - // [[ val x = { def f = ...; let y = ...; STMT }; STMT ]] = def f = ...; let y = ...; val x = STMT; STMT - case Stmt.Scope(ds, bodyBinding) => - normal.Scope(ds, normalizeVal(id, tpe, bodyBinding, body)) + // [[ val x = { let y = ...; STMT }; STMT ]] = let y = ...; val x = STMT; STMT + case Stmt.Let(id2, tpe2, binding2, body2) => + Stmt.Let(id2, tpe2, binding2, normalizeVal(id, tpe, body2, body)) // Flatten vals. This should be non-leaking since we use garbage free refcounting. // [[ val x = { val y = stmt1; stmt2 }; stmt3 ]] = [[ val y = stmt1; val x = stmt2; stmt3 ]] case Stmt.Val(id2, tpe2, binding2, body2) => normalizeVal(id2, tpe2, binding2, Stmt.Val(id, tpe, body2, body)) - // [[ val x = { var y in r = e; stmt1 }; stmt2 ]] = var y in r = e; [[ val x = stmt1; stmt2 ]] case Stmt.Alloc(id2, init2, region2, body2) => Stmt.Alloc(id2, init2, region2, normalizeVal(id, tpe, body2, body)) @@ -259,7 +268,10 @@ object Normalizer { normal => case b @ Block.BlockLit(tparams, cparams, vparams, bparams, body) => normalize(b) // [[ unbox (box b) ]] = [[ b ]] - case Block.Unbox(pure) => normal.Unbox(normalize(pure)) + case Block.Unbox(pure) => normalize(pure) match { + case Pure.Box(b, _) => b + case p => Block.Unbox(p) + } case Block.New(impl) => New(normalize(impl)) } @@ -283,74 +295,25 @@ object Normalizer { normal => // [[ box (unbox e) ]] = [[ e ]] case Pure.Box(b, annotatedCapture) => active(b) match { case NormalizedBlock.Known(Unbox(p), boundBy) => p - case _ => normal.Box(normalize(b), annotatedCapture) + case _ => normalize(b) match { + case Block.Unbox(pure) => pure + case b => Pure.Box(b, annotatedCapture) + } } // congruences - case Pure.PureApp(b, targs, vargs) => Pure.PureApp(normalize(b), targs, vargs.map(normalize)) + case Pure.PureApp(f, targs, vargs) => Pure.PureApp(f, targs, vargs.map(normalize)) case Pure.Make(data, tag, vargs) => Pure.Make(data, tag, vargs.map(normalize)) case Pure.ValueVar(id, annotatedType) => p case Pure.Literal(value, annotatedType) => p } def normalize(e: Expr)(using Context): Expr = e match { - case DirectApp(b, targs, vargs, bargs) => DirectApp(normalize(b), targs, vargs.map(normalize), bargs.map(normalize)) - - // [[ run (return e) ]] = [[ e ]] - case Run(s) => normal.Run(normalize(s)) + case DirectApp(b, targs, vargs, bargs) => DirectApp(b, targs, vargs.map(normalize), bargs.map(normalize)) case pure: Pure => normalize(pure) } - - // Smart Constructors - // ------------------ - @tailrec - private def Scope(definitions: List[Definition], body: Stmt): Stmt = body match { - - // flatten scopes - // { def f = ...; { def g = ...; BODY } } = { def f = ...; def g; BODY } - case Stmt.Scope(others, body) => normal.Scope(definitions ++ others, body) - - // commute bindings - // let x = run { let y = e; s } = let y = e; let x = run { s } - case _ => if (definitions.isEmpty) body else { - var defsSoFar: List[Definition] = Nil - - definitions.foreach { - case Definition.Let(id, tpe, Run(Stmt.Scope(ds, body))) => - defsSoFar = defsSoFar ++ (ds :+ Definition.Let(id, tpe, normal.Run(body))) - case d => defsSoFar = defsSoFar :+ d - } - Stmt.Scope(defsSoFar, body) - } - } - - private def Run(s: Stmt): Expr = s match { - - // run { let x = e; return x } = e - case Stmt.Scope(Definition.Let(id1, _, binding) :: Nil, Stmt.Return(Pure.ValueVar(id2, _))) if id1 == id2 => - binding - - // run { return e } = e - case Stmt.Return(expr) => expr - - case _ => core.Run(s) - } - - // box (unbox p) = p - private def Box(b: Block, capt: Captures): Pure = b match { - case Block.Unbox(pure) => pure - case b => Pure.Box(b, capt) - } - - // unbox (box b) = b - private def Unbox(p: Pure): Block = p match { - case Pure.Box(b, _) => b - case p => Block.Unbox(p) - } - - // Helpers for beta-reduction // -------------------------- @@ -364,20 +327,19 @@ object Normalizer { normal => // Only bind if not already a variable!!! var ids: Set[Id] = Set.empty - var bindings: List[Definition.Def] = Nil + var bindings: List[Binding] = Nil var bvars: List[Block.BlockVar] = Nil // (1) first bind (b.bparams zip bargs) foreach { case (bparam, x: Block.BlockVar) => - // Update usage: u1 + (u2 - 1) usage.update(x.id, usage.getOrElse(bparam.id, Usage.Never) + usage.getOrElse(x.id, Usage.Never).decrement) bvars = bvars :+ x // introduce a binding case (bparam, block) => val id = symbols.TmpBlock("blockBinding") - bindings = bindings :+ Definition.Def(id, block) + bindings = bindings :+ Binding.Def(id, block) bvars = bvars :+ Block.BlockVar(id, block.tpe, block.capt) copyUsage(bparam.id, id) ids += id @@ -392,7 +354,7 @@ object Normalizer { normal => // (2) substitute val body = substitutions.substitute(renamedLit, targs, vargs, bvars) - normalize(normal.Scope(bindings, body)) + normalize(Binding(bindings, body)) } private def selectOperation(impl: Implementation, method: Id): Block.BlockLit = diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala index f0eb439ed..5ea5e008e 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala @@ -11,14 +11,19 @@ class Reachable( var seen: Set[Id] ) { + // TODO we could use [[Binding]] here. + type Definitions = Map[Id, Block | Expr | Stmt] + private def update(id: Id, u: Usage): Unit = reachable = reachable.updated(id, u) private def usage(id: Id): Usage = reachable.getOrElse(id, Usage.Never) - def process(d: Definition)(using defs: Map[Id, Definition]): Unit = - if stack.contains(d.id) then update(d.id, Usage.Recursive) - else d match { - case Definition.Def(id, block) => - seen = seen + id + def processDefinition(id: Id, d: Block | Expr | Stmt)(using defs: Definitions): Unit = { + if stack.contains(id) then { update(id, Usage.Recursive); return } + + seen = seen + id + + d match { + case block: Block => val before = stack stack = id :: stack @@ -26,13 +31,12 @@ class Reachable( process(block) stack = before - case Definition.Let(id, _, binding) => - seen = seen + id - - process(binding) + case binding: Expr => process(binding) + case binding: Stmt => process(binding) } + } - def process(id: Id)(using defs: Map[Id, Definition]): Unit = + def process(id: Id)(using defs: Definitions): Unit = if (stack.contains(id)) { update(id, Usage.Recursive) return; @@ -41,10 +45,12 @@ class Reachable( update(id, usage(id) + Usage.Once) if (!seen.contains(id)) { - defs.get(id).foreach(process) + seen = seen + id + defs.get(id).foreach { d => processDefinition(id, d) } } - def process(b: Block)(using defs: Map[Id, Definition]): Unit = + + def process(b: Block)(using defs: Definitions): Unit = b match { case Block.BlockVar(id, annotatedTpe, annotatedCapt) => process(id) case Block.BlockLit(tparams, cparams, vparams, bparams, body) => process(body) @@ -52,22 +58,15 @@ class Reachable( case Block.New(impl) => process(impl) } - def process(s: Stmt)(using defs: Map[Id, Definition]): Unit = s match { - case Stmt.Scope(definitions, body) => - var currentDefs = defs - definitions.foreach { - case d: Definition.Def => - currentDefs += d.id -> d // recursive - // Do NOT process them here, since this would mean the definition is used - // process(d)(using currentDefs) - case d: Definition.Let => - // DO only process if NOT pure - if (d.binding.capt.nonEmpty) { - process(d)(using currentDefs) - } - currentDefs += d.id -> d // non-recursive - } - process(body)(using currentDefs) + def process(s: Stmt)(using defs: Definitions): Unit = s match { + case Stmt.Def(id, block, body) => + // Do NOT process `block` here, since this would mean the definition is used + process(body)(using defs + (id -> block)) + case Stmt.Let(id, tpe, binding, body) => + // DO only process if impure, since we need to keep it in this case + // for its side effects + if (binding.capt.nonEmpty) { processDefinition(id, binding) } + process(body)(using defs + (id -> binding)) case Stmt.Return(expr) => process(expr) case Stmt.Val(id, tpe, binding, body) => process(binding); process(body) case Stmt.App(callee, targs, vargs, bargs) => @@ -100,12 +99,11 @@ class Reachable( case Stmt.Hole() => () } - def process(e: Expr)(using defs: Map[Id, Definition]): Unit = e match { + def process(e: Expr)(using defs: Definitions): Unit = e match { case DirectApp(b, targs, vargs, bargs) => - process(b); + process(b) vargs.foreach(process) bargs.foreach(process) - case Run(s) => process(s) case Pure.ValueVar(id, annotatedType) => process(id) case Pure.Literal(value, annotatedType) => () case Pure.PureApp(b, targs, vargs) => process(b); vargs.foreach(process) @@ -114,13 +112,16 @@ class Reachable( case Pure.Box(b, annotatedCapture) => process(b) } - def process(i: Implementation)(using defs: Map[Id, Definition]): Unit = + def process(i: Implementation)(using defs: Definitions): Unit = i.operations.foreach { op => process(op.body) } } object Reachable { def apply(entrypoints: Set[Id], m: ModuleDecl): Map[Id, Usage] = { - val definitions = m.definitions.map(d => d.id -> d).toMap + val definitions: Map[Id, Block | Expr | Stmt] = m.definitions.map { + case Toplevel.Def(id, block) => id -> block + case Toplevel.Val(id, tpe, binding) => id -> binding + }.toMap val initialUsage = entrypoints.map { id => id -> Usage.Recursive }.toMap val analysis = new Reachable(initialUsage, Nil, Set.empty) diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/RemoveTailResumptions.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/RemoveTailResumptions.scala index bf6544a70..2174f7acd 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/RemoveTailResumptions.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/RemoveTailResumptions.scala @@ -18,10 +18,11 @@ object RemoveTailResumptions { def tailResumptive(k: Id, stmt: Stmt): Boolean = def freeInStmt(stmt: Stmt): Boolean = Variables.free(stmt).containsBlock(k) def freeInExpr(expr: Expr): Boolean = Variables.free(expr).containsBlock(k) - def freeInDef(definition: Definition): Boolean = Variables.free(definition).containsBlock(k) + def freeInBlock(block: Block): Boolean = Variables.free(block).containsBlock(k) stmt match { - case Stmt.Scope(definitions, body) => definitions.forall { d => !freeInDef(d) } && tailResumptive(k, body) + case Stmt.Def(id, block, body) => !freeInBlock(block) && tailResumptive(k, body) + case Stmt.Let(id, tpe, binding, body) => !freeInExpr(binding) && tailResumptive(k, body) case Stmt.Return(expr) => false case Stmt.Val(id, annotatedTpe, binding, body) => tailResumptive(k, body) && !freeInStmt(binding) case Stmt.App(callee, targs, vargs, bargs) => false @@ -43,7 +44,8 @@ object RemoveTailResumptions { } def removeTailResumption(k: Id, stmt: Stmt): Stmt = stmt match { - case Stmt.Scope(definitions, body) => Stmt.Scope(definitions, removeTailResumption(k, body)) + case Stmt.Def(id, block, body) => Stmt.Def(id, block, removeTailResumption(k, body)) + case Stmt.Let(id, tpe, binding, body) => Stmt.Let(id, tpe, binding, removeTailResumption(k, body)) case Stmt.Val(id, annotatedTpe, binding, body) => Stmt.Val(id, annotatedTpe, binding, removeTailResumption(k, body)) case Stmt.If(cond, thn, els) => Stmt.If(cond, removeTailResumption(k, thn), removeTailResumption(k, els)) case Stmt.Match(scrutinee, clauses, default) => Stmt.Match(scrutinee, clauses.map { diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/StaticArguments.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/StaticArguments.scala index 3a8e1214b..c6729192b 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/StaticArguments.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/StaticArguments.scala @@ -59,7 +59,7 @@ object StaticArguments { * foo_worker(a_fresh, b_fresh) * foo(4, 2, 0) */ - def wrapDefinition(id: Id, blockLit: BlockLit)(using ctx: StaticArgumentsContext): Definition.Def = + def wrapDefinition(id: Id, blockLit: BlockLit)(using ctx: StaticArgumentsContext): BlockLit = { val IsStatic(staticT, staticV, staticB) = ctx.statics(id) assert(staticT.forall(x => x), "Can only apply the static arguments translation, if all type arguments are static.") @@ -94,40 +94,38 @@ object StaticArguments { val workerVar: Block.BlockVar = BlockVar(Id(id.name.name + "_worker"), workerType, newCapture) ctx.workers(id) = workerVar - Definition.Def(id, BlockLit( + BlockLit( blockLit.tparams, freshCparams, freshVparams, freshBparams, - Scope(List(Definition.Def(workerVar.id, BlockLit( + Stmt.Def(workerVar.id, BlockLit( dropStatic(staticT, blockLit.tparams), dropStatic(staticB, blockLit.cparams), dropStatic(staticV, blockLit.vparams), dropStatic(staticB, blockLit.bparams), rewrite(blockLit.body)(using enterFunction(id)) - ))), App( + ), App( workerVar, dropStatic(staticT, blockLit.tparams.map(t => ValueType.Var(t))), dropStatic(staticV, freshVparams.map(v => ValueVar(v.id, v.tpe))), dropStatic(staticB, freshBparams.map(b => BlockVar(b.id, b.tpe, b.capt))) )) - )) - - def rewrite(definitions: List[Definition])(using ctx: StaticArgumentsContext): List[Definition] = - definitions.collect { - case Definition.Def(id, block: BlockLit) if hasStatics(id) => wrapDefinition(id, block) - case Definition.Def(id, block) => Definition.Def(id, rewrite(block)) - case Definition.Let(id, tpe, binding) => Definition.Let(id, tpe, rewrite(binding)) - } + ) + } - def rewrite(d: Definition)(using StaticArgumentsContext): Definition = d match { - case Definition.Def(id, block) => Definition.Def(id, rewrite(block)) - case Definition.Let(id, tpe, binding) => Definition.Let(id, tpe, rewrite(binding)) + def rewrite(d: Toplevel)(using StaticArgumentsContext): Toplevel = d match { + case Toplevel.Def(id, block: BlockLit) if hasStatics(id) => Toplevel.Def(id, wrapDefinition(id, block)) + case Toplevel.Def(id, block) => Toplevel.Def(id, rewrite(block)) + case Toplevel.Val(id, tpe, binding) => Toplevel.Val(id, tpe, rewrite(binding)) } def rewrite(s: Stmt)(using C: StaticArgumentsContext): Stmt = s match { - case Stmt.Scope(definitions, body) => - MaybeScope(rewrite(definitions), rewrite(body)) + + case Stmt.Def(id, block: BlockLit, body) if hasStatics(id) => Stmt.Def(id, wrapDefinition(id, block), rewrite(body)) + case Stmt.Def(id, block, body) => Stmt.Def(id, rewrite(block), rewrite(body)) + + case Stmt.Let(id, tpe, binding, body) => Stmt.Let(id, tpe, rewrite(binding), rewrite(body)) case Stmt.App(b, targs, vargs, bargs) => b match { @@ -186,7 +184,7 @@ object StaticArguments { } def rewrite(p: Pure)(using StaticArgumentsContext): Pure = p match { - case Pure.PureApp(b, targs, vargs) => Pure.PureApp(rewrite(b), targs, vargs.map(rewrite)) + case Pure.PureApp(f, targs, vargs) => Pure.PureApp(f, targs, vargs.map(rewrite)) case Pure.Make(data, tag, vargs) => Pure.Make(data, tag, vargs.map(rewrite)) case x @ Pure.ValueVar(id, annotatedType) => x @@ -197,10 +195,9 @@ object StaticArguments { } def rewrite(e: Expr)(using StaticArgumentsContext): Expr = e match { - case DirectApp(b, targs, vargs, bargs) => DirectApp(rewrite(b), targs, vargs.map(rewrite), bargs.map(rewrite)) + case DirectApp(b, targs, vargs, bargs) => DirectApp(b, targs, vargs.map(rewrite), bargs.map(rewrite)) // congruences - case Run(s) => Run(rewrite(s)) case pure: Pure => rewrite(pure) } @@ -236,7 +233,7 @@ object StaticArguments { List() ) - val updatedDefs = rewrite(m.definitions) + val updatedDefs = m.definitions.map(rewrite) m.copy(definitions = updatedDefs) } diff --git a/effekt/shared/src/main/scala/effekt/cps/Transformer.scala b/effekt/shared/src/main/scala/effekt/cps/Transformer.scala index a1e1d6cc9..3f2422fe5 100644 --- a/effekt/shared/src/main/scala/effekt/cps/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/cps/Transformer.scala @@ -26,14 +26,12 @@ object Transformer { ModuleDecl(path, includes, declarations, externs.map(transform), definitions.map(transformToplevel), exports) } - def transformToplevel(definition: core.Definition)(using TransformationContext): ToplevelDefinition = definition match { - case core.Definition.Def(id, block) => ToplevelDefinition.Def(id, transform(block)) - case core.Definition.Let(id, tpe, core.Run(stmt)) => + def transformToplevel(definition: core.Toplevel)(using TransformationContext): ToplevelDefinition = definition match { + case core.Toplevel.Def(id, block) => ToplevelDefinition.Def(id, transform(block)) + case core.Toplevel.Val(id, tpe, stmt) => val ks = Id("ks") val k = Id("k") ToplevelDefinition.Val(id, ks, k, transform(stmt, ks, Continuation.Dynamic(k))) - case core.Definition.Let(id, tpe, binding: core.DirectApp) => sys error "Not supported" - case core.Definition.Let(id, tpe, p: core.Pure) => ToplevelDefinition.Let(id, transform(p)) } def transform(extern: core.Extern)(using TransformationContext): Extern = extern match { @@ -48,44 +46,29 @@ object Transformer { case core.ExternBody.Unsupported(err) => ExternBody.Unsupported(err) } - def transform(stmt: core.Stmt, ks: Id, k: Continuation)(using TransformationContext): Stmt = stmt match { - case core.Stmt.Scope(definitions, body) => + def transform(stmt: core.Stmt, ks: Id, k: Continuation)(using C: TransformationContext): Stmt = stmt match { - def go(definitions: List[core.Definition], acc: List[Def], ks: Id, k: Continuation)(using C: TransformationContext): Stmt = - definitions match { - case Nil => bindDefs(acc, transform(body, ks, k)) + // dealiasing + case core.Stmt.Def(id, core.BlockVar(x, _, _), body) => + binding(id, C.lookupBlock(x)) { transform(body, ks, k) } - // dealiasing - case core.Definition.Def(id, core.BlockVar(x, _, _)) :: xs => - binding(id, C.lookupBlock(x)) { go(xs, acc, ks, k) } + case core.Stmt.Def(id, block, body) => + LetDef(id, transform(block), transform(body, ks, k)) - case core.Definition.Def(id, block) :: xs => - go(xs, acc :+ Def(id, transform(block)), ks, k) + // dealiasing + case core.Stmt.Let(id, tpe, core.Pure.ValueVar(x, _), body) => + binding(id, C.lookupValue(x)) { transform(body, ks, k) } - case core.Definition.Let(id, tpe, core.Run(stmt)) :: xs => - bindDefs(acc, transform(stmt, ks, Continuation.Static(id) { (value, ks) => - binding(id, value) { go(xs, Nil, ks, k) } - })) - - case core.Definition.Let(id, tpe, core.DirectApp(b, targs, vargs, bargs)) :: xs => - transform(b) match { - case Block.BlockVar(f) => - bindDefs(acc, LetExpr(id, DirectApp(f, vargs.map(transform), bargs.map(transform)), - go(xs, Nil, ks, k))) - case _ => sys error "Should not happen" - } - - // dealias - case core.Definition.Let(id, tpe, core.Pure.ValueVar(x, _)) :: xs => - binding(id, C.lookupValue(x)) { go(xs, acc, ks, k) } - - case core.Definition.Let(id, tpe, pure: core.Pure) :: xs => - bindDefs(acc, LetExpr(id, transform(pure), go(xs, Nil, ks, k))) - } - - def bindDefs(acc: List[Def], body: Stmt): Stmt = if acc.isEmpty then body else Scope(acc, body) + case core.Stmt.Let(id, tpe, core.DirectApp(b, targs, vargs, bargs), body) => + transform(b) match { + case Block.BlockVar(f) => + LetExpr(id, DirectApp(f, vargs.map(transform), bargs.map(transform)), + transform(body, ks, k)) + case _ => sys error "Should not happen" + } - go(definitions, Nil, ks, k) + case core.Stmt.Let(id, tpe, pure: core.Pure, body) => + LetExpr(id, transform(pure), transform(body, ks, k)) case core.Stmt.Return(value) => k(transform(value), ks) diff --git a/effekt/shared/src/main/scala/effekt/cps/Tree.scala b/effekt/shared/src/main/scala/effekt/cps/Tree.scala index a2dbb18c9..774a6a4f2 100644 --- a/effekt/shared/src/main/scala/effekt/cps/Tree.scala +++ b/effekt/shared/src/main/scala/effekt/cps/Tree.scala @@ -103,9 +103,6 @@ enum Stmt extends Tree { case Jump(k: Id, arg: Pure, ks: MetaCont) // if the continuation is known, we inline and don't jump - // these could in principle be mutually recursive - case Scope(definitions: List[Def], body: Stmt) - case App(callee: Block, vargs: List[Pure], bargs: List[Block], ks: MetaCont, k: Cont) case Invoke(callee: Block, method: Id, vargs: List[Pure], bargs: List[Block], ks: MetaCont, k: Cont) @@ -114,6 +111,7 @@ enum Stmt extends Tree { case If(cond: Pure, thn: Stmt, els: Stmt) case Match(scrutinee: Pure, clauses: List[(Id, Clause)], default: Option[Stmt]) + case LetDef(id: Id, binding: Block, body: Stmt) case LetExpr(id: Id, binding: Expr, body: Stmt) case LetCont(id: Id, binding: Cont.ContLam, body: Stmt) @@ -203,12 +201,11 @@ object Variables { } def free(s: Stmt): Variables = s match { case Stmt.Jump(k, arg, ks) => cont(k) ++ free(arg) ++ free(ks) - case Stmt.Scope(definitions, body) => - all(definitions, free) ++ free(body) -- all(definitions, d => block(d.id)) case Stmt.App(callee, vargs, bargs, ks, k) => free(callee) ++ all(vargs, free) ++ all(bargs, free) ++ free(ks) ++ free(k) case Stmt.Invoke(callee, method, vargs, bargs, ks, k) => free(callee) ++ all(vargs, free) ++ all(bargs, free) ++ free(ks) ++ free(k) case Stmt.If(cond, thn, els) => free(cond) ++ free(thn) ++ free(els) case Stmt.Match(scrutinee, clauses, default) => free(scrutinee) ++ all(clauses, free) ++ all(default, free) + case Stmt.LetDef(id, binding, body) => (free(binding) ++ free(body)) -- block(id) case Stmt.LetExpr(id, binding, body) => free(binding) ++ (free(body) -- value(id)) case Stmt.LetCont(id, binding, body) => free(binding) ++ (free(body) -- cont(id)) diff --git a/effekt/shared/src/main/scala/effekt/generator/chez/Transformer.scala b/effekt/shared/src/main/scala/effekt/generator/chez/Transformer.scala index 968b9d707..f72d6d35e 100644 --- a/effekt/shared/src/main/scala/effekt/generator/chez/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/generator/chez/Transformer.scala @@ -60,17 +60,14 @@ trait Transformer { chez.Block(generateStateAccessors(pure) ++ definitions, Nil, runMain(nameRef(mainSymbol))) } - def toChez(p: Param): ChezName = nameDef(p.id) + def toChez(p: ValueParam): ChezName = nameDef(p.id) + def toChez(p: BlockParam): ChezName = nameDef(p.id) def toChez(module: ModuleDecl)(using ErrorReporter): List[chez.Def] = { val decls = module.declarations.flatMap(toChez) val externs = module.externs.map(toChez) // TODO FIXME, once there is a let _ = ... in there, we are doomed! - val defns = module.definitions.map(toChez).flatMap { - case Left(d) => Some(d) - case Right(None) => None - case Right(e) => ??? - } + val defns = module.definitions.map(toChez) decls ++ externs ++ defns } @@ -121,7 +118,8 @@ trait Transformer { case Region(body) => chez.Builtin("with-region", toChez(body)) - case s: Scope => chez.Let(Nil, toChez(s)) + case stmt: (Def | Let) => + chez.Let(Nil, toChez(stmt)) } def toChez(decl: core.Declaration): List[chez.Def] = decl match { @@ -142,7 +140,7 @@ trait Transformer { chez.Builtin("hole") } chez.Constant(nameDef(id), - chez.Lambda((vps ++ bps) map { p => nameDef(p.id) }, + chez.Lambda(vps.map { p => nameDef(p.id) } ++ bps.map { p => nameDef(p.id) }, tBody)) case Extern.Include(ff, contents) => @@ -152,40 +150,39 @@ trait Transformer { def toChez(t: Template[core.Expr]): chez.Expr = chez.RawExpr(t.strings, t.args.map(e => toChez(e))) - def toChez(defn: Definition): Either[chez.Def, Option[chez.Expr]] = defn match { - case Definition.Def(id, block) => - Left(chez.Constant(nameDef(id), toChez(block))) + def toChez(defn: Toplevel): chez.Def = defn match { + case Toplevel.Def(id, block) => chez.Constant(nameDef(id), toChez(block)) + case Toplevel.Val(id, tpe, binding) => chez.Constant(nameDef(id), run(toChezExpr(binding))) + } - case Definition.Let(Wildcard(), _, binding) => + def toChez(stmt: Stmt): chez.Block = stmt match { + case Stmt.Def(id, block, body) => + val chez.Block(defs, exprs, result) = toChez(body) + chez.Block(chez.Constant(nameDef(id), toChez(block)) :: defs, exprs, result) + + case Stmt.Let(Wildcard(), tpe, binding, body) => toChez(binding) match { // drop the binding altogether, if it is of the form: // let _ = myVariable; BODY // since this might lead to invalid scheme code. - case _: chez.Variable => Right(None) - case other => Right(Some(other)) + case _: chez.Variable => toChez(body) + case expr => + toChez(body) match { + case chez.Block(Nil, exprs, result) => chez.Block(Nil, expr :: exprs, result) + case rest => chez.Block(Nil, expr :: Nil, chez.Let(Nil, rest)) + } } - // we could also generate a let here... - case Definition.Let(id, _, binding) => - Left(chez.Constant(nameDef(id), toChez(binding))) - } - - def toChez(stmt: Stmt): chez.Block = stmt match { - // TODO maybe this can be simplified after also introducing mutual definitions - case Scope(definitions, body) => - definitions.map(toChez).foldRight(toChez(body)) { - case (Left(defn), chez.Block(defns, exprs, result)) => chez.Block(defn :: defns, exprs, result) - case (Right(Some(expr)), chez.Block(Nil, exprs, result)) => chez.Block(Nil, expr :: exprs, result) - case (Right(Some(expr)), rest) => chez.Block(Nil, expr :: Nil, chez.Let(Nil, rest)) - case (Right(None), rest) => rest - } + case Stmt.Let(id, tpe, binding, body) => + val chez.Block(defs, exprs, result) = toChez(body) + chez.Block(chez.Constant(nameDef(id), toChez(binding)) :: defs, exprs, result) case other => chez.Block(Nil, Nil, toChezExpr(other)) } def toChez(block: BlockLit): chez.Lambda = block match { case BlockLit(tps, cps, vps, bps, body) => - chez.Lambda((vps ++ bps) map toChez, toChez(body)) + chez.Lambda(vps.map(toChez) ++ bps.map(toChez), toChez(body)) } def toChez(block: Block): chez.Expr = block match { @@ -205,7 +202,7 @@ trait Transformer { def toChez(op: Operation): chez.Expr = op match { case Operation(name, tps, cps, vps, bps, body) => - chez.Lambda((vps ++ bps) map toChez, toChez(body)) + chez.Lambda(vps.map(toChez) ++ bps.map(toChez), toChez(body)) } def toChez(expr: Expr): chez.Expr = expr match { @@ -224,8 +221,6 @@ trait Transformer { chez.Call(nameRef(field), toChez(b)) case Box(b, _) => toChez(b) - - case Run(s) => run(toChezExpr(s)) } diff --git a/effekt/shared/src/main/scala/effekt/generator/js/Transformer.scala b/effekt/shared/src/main/scala/effekt/generator/js/Transformer.scala index 6675c3ef3..f0065b61b 100644 --- a/effekt/shared/src/main/scala/effekt/generator/js/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/generator/js/Transformer.scala @@ -168,7 +168,7 @@ trait Transformer { } // Traverse tree once more to find all used symbols, defined in other modules. - def findUsedDependencies(t: Definition) = + def findUsedDependencies(t: Toplevel) = def go(t: Any): Unit = Tree.visit(t) { case BlockVar(x, tpe, capt) if publicDependencySymbols.isDefinedAt(x) => register(publicDependencySymbols(x), x) diff --git a/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala b/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala index 7478c41dd..a796cd5bb 100644 --- a/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala +++ b/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala @@ -166,9 +166,9 @@ object TransformerCps extends Transformer { } def toJS(s: cps.Stmt)(using D: TransformerContext): Binding[js.Stmt] = s match { - case cps.Stmt.Scope(defs, body) => + case cps.Stmt.LetDef(id, block, body) => Binding { k => - defs.map(toJS) ++ toJS(body).run(k) + js.Const(nameDef(id), requiringThunk { toJS(block) }) :: toJS(body).run(k) } case cps.Stmt.If(cond, thn, els) => diff --git a/effekt/shared/src/main/scala/effekt/machine/Transformer.scala b/effekt/shared/src/main/scala/effekt/machine/Transformer.scala index 0e010bd25..798d6ced2 100644 --- a/effekt/shared/src/main/scala/effekt/machine/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/machine/Transformer.scala @@ -2,7 +2,7 @@ package effekt package machine import effekt.context.Context -import effekt.core.{ Block, DeclarationContext, Definition, Id, given } +import effekt.core.{ Block, DeclarationContext, Toplevel, Id, given } import effekt.symbols.{ Symbol, TermSymbol } import effekt.symbols.builtins.TState import effekt.util.messages.ErrorReporter @@ -33,10 +33,10 @@ object Transformer { findToplevelBlocksParams(definitions) val transformedDefinitions = definitions.foldLeft(mainEntry) { - case (rest, core.Definition.Def(id, core.BlockLit(tparams, cparams, vparams, bparams, body))) => + case (rest, core.Toplevel.Def(id, core.BlockLit(tparams, cparams, vparams, bparams, body))) => Def(Label(transform(id), vparams.map(transform) ++ bparams.map(transform)), transform(body), rest) - case (rest, core.Definition.Let(id, tpe, binding)) => - Def(BC.globals(id), transform(binding).run { value => Return(List(value)) }, rest) + case (rest, core.Toplevel.Val(id, tpe, binding)) => + Def(BC.globals(id), transform(binding), rest) case (rest, d) => ErrorReporter.abort(s"Toplevel object definitions not yet supported: ${d}") } @@ -68,90 +68,73 @@ object Transformer { def transform(stmt: core.Stmt)(using BPC: BlocksParamsContext, DC: DeclarationContext, E: ErrorReporter): Statement = stmt match { - case core.Scope(definitions, rest) => - // (1) Collect all the information about free variables of local definitions - definitions.foreach { - case Definition.Def(id, block @ core.BlockLit(tparams, cparams, vparams, bparams, body)) => - - noteParameters(bparams) - - // TODO use existing lambda lifting code - - // Does not work for mutually recursive local definitions (which are not supported anyway, at the moment) - val freeVariables = core.Variables.free(block).toSet - .filterNot(x => BPC.globals.contains(x.id)) // globals are NOT free - - val freeParams = freeVariables.flatMap { - case core.Variable.Value(id, tpe) => - Set(Variable(transform(id), transform(tpe))) - // Mutable variables are blocks and can be free, but do not have info. - case core.Variable.Block(id, core.Type.TState(stTpe), capt) => - Set(Variable(transform(id), Type.Reference(transform(stTpe)))) + case core.Def(id, block @ core.BlockLit(tparams, cparams, vparams, bparams, body), rest) => + // (1) Collect all the information about free variables of local definitions + noteParameters(bparams) - // Regions are blocks and can be free, but do not have info. - case core.Variable.Block(id, core.Type.TRegion, capt) => - if id == symbols.builtins.globalRegion - then Set.empty - else Set(Variable(transform(id), Type.Prompt())) + // Does not work for mutually recursive local definitions (which are not supported anyway, at the moment) + val freeVariables = core.Variables.free(block).toSet + .filterNot(x => BPC.globals.contains(x.id)) // globals are NOT free - case core.Variable.Block(pid, tpe, capt) if pid != id => BPC.info.get(pid) match { - // For each known free block we have to add its free variables to this one (flat closure) - case Some(BlockInfo.Definition(freeParams, blockParams)) => - freeParams.toSet + val freeParams = freeVariables.flatMap { + case core.Variable.Value(id, tpe) => + Set(Variable(transform(id), transform(tpe))) - // Unknown free blocks stay free variables - case Some(BlockInfo.Parameter(tpe)) => - Set(Variable(transform(pid), transform(tpe))) + // Mutable variables are blocks and can be free, but do not have info. + case core.Variable.Block(id, core.Type.TState(stTpe), capt) => + Set(Variable(transform(id), Type.Reference(transform(stTpe)))) - // Everything else is considered bound or global - case None => - ErrorReporter.panic(s"Could not find info for free variable $pid") - } - case _ => Set.empty - } + // Regions are blocks and can be free, but do not have info. + case core.Variable.Block(id, core.Type.TRegion, capt) => + if id == symbols.builtins.globalRegion + then Set.empty + else Set(Variable(transform(id), Type.Prompt())) - noteDefinition(id, vparams.map(transform) ++ bparams.map(transform), freeParams.toList) + case core.Variable.Block(pid, tpe, capt) if pid != id => BPC.info.get(pid) match { + // For each known free block we have to add its free variables to this one (flat closure) + case Some(BlockInfo.Definition(freeParams, blockParams)) => + freeParams.toSet - case Definition.Def(id, block @ core.New(impl)) => - // this is just a hack... - noteParameter(id, block.tpe) + // Unknown free blocks stay free variables + case Some(BlockInfo.Parameter(tpe)) => + Set(Variable(transform(pid), transform(tpe))) - case Definition.Def(id, core.BlockVar(alias, tpe, _)) => - getDefinition(alias) match { - case BlockInfo.Definition(free, params) => - noteDefinition(id, free, params) + // Everything else is considered bound or global + case None => + ErrorReporter.panic(s"Could not find info for free variable $pid") } - - case Definition.Def(id, b @ core.Unbox(_)) => - noteParameter(id, b.tpe) - - case Definition.Let(_, _, _) => - () + case _ => Set.empty } + noteDefinition(id, vparams.map(transform) ++ bparams.map(transform), freeParams.toList) + // (2) Actually translate the definitions - definitions.foldRight(transform(rest)) { - case (core.Definition.Let(id, tpe, binding), rest) => - transform(binding).run { value => - // TODO consider passing the environment to [[transform]] instead of explicit substitutions here. - // TODO it is important that we use the inferred [[binding.tpe]] and not the annotated type [[tpe]], but why? - Substitute(List(Variable(transform(id), transform(binding.tpe)) -> value), rest) - } + Def(transformLabel(id), transform(body), transform(rest)) - case (core.Definition.Def(id, core.BlockLit(tparams, cparams, vparams, bparams, body)), rest) => - Def(transformLabel(id), transform(body), rest) + case core.Def(id, block @ core.New(impl), rest) => + // this is just a hack... + noteParameter(id, block.tpe) + New(Variable(transform(id), transform(impl.interface)), transform(impl), transform(rest)) - case (core.Definition.Def(id, core.New(impl)), rest) => - New(Variable(transform(id), transform(impl.interface)), transform(impl), rest) + case core.Def(id, block @ core.BlockVar(alias, tpe, _), rest) => + getDefinition(alias) match { + case BlockInfo.Definition(free, params) => + noteDefinition(id, free, params) + } + Def(transformLabel(id), Jump(transformLabel(alias)), transform(rest)) - case (core.Definition.Def(id, core.BlockVar(alias, tpe, _)), rest) => - Def(transformLabel(id), Jump(transformLabel(alias)), rest) + case core.Def(id, block @ core.Unbox(pure), rest) => + noteParameter(id, block.tpe) + transform(pure).run { boxed => + ForeignCall(Variable(transform(id), Type.Negative()), "unbox", List(boxed), transform(rest)) + } - case (core.Definition.Def(id, core.Unbox(pure)), rest) => - transform(pure).run { boxed => - ForeignCall(Variable(transform(id), Type.Negative()), "unbox", List(boxed), rest) - } + case core.Let(id, tpe, binding, rest) => + transform(binding).run { value => + // TODO consider passing the environment to [[transform]] instead of explicit substitutions here. + // TODO it is important that we use the inferred [[binding.tpe]] and not the annotated type [[tpe]], but why? + Substitute(List(Variable(transform(id), transform(binding.tpe)) -> value), transform(rest)) } case core.Return(expr) => @@ -453,13 +436,6 @@ object Transformer { } } - case core.Run(stmt) => - // NOTE: `stmt` is guaranteed to be of type `tpe`. - val variable = Variable(freshName("run"), transform(stmt.tpe)) - Binding { k => - PushFrame(Clause(List(variable), k(variable)), transform(stmt)) - } - case core.Box(block, annot) => transformBlockArg(block).flatMap { unboxed => Binding { k => @@ -519,7 +495,7 @@ object Transformer { case core.BlockType.Interface(symbol, targs) => Negative() } - def transformLabel(id: Id)(using BPC: BlocksParamsContext) = getDefinition(id) match { + def transformLabel(id: Id)(using BPC: BlocksParamsContext): Label = getDefinition(id) match { case BlockInfo.Definition(freeParams, boundParams) => Label(transform(id), boundParams ++ freeParams) } @@ -537,12 +513,12 @@ object Transformer { def freshName(baseName: String): String = baseName + "_" + symbols.Symbol.fresh.next() - def findToplevelBlocksParams(definitions: List[core.Definition])(using BlocksParamsContext, ErrorReporter): Unit = + def findToplevelBlocksParams(definitions: List[core.Toplevel])(using BlocksParamsContext, ErrorReporter): Unit = definitions.foreach { - case Definition.Def(id, core.BlockLit(tparams, cparams, vparams, bparams, body)) => + case Toplevel.Def(id, core.BlockLit(tparams, cparams, vparams, bparams, body)) => noteDefinition(id, vparams.map(transform) ++ bparams.map(transform), Nil) noteParameters(bparams) - case Definition.Let(id, tpe, binding) => + case Toplevel.Val(id, tpe, binding) => noteDefinition(id, Nil, Nil) noteGlobal(id) case other => () From 87a84a8636640defe66b31d8560f46961331abbb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jonathan=20Immanuel=20Brachth=C3=A4user?= Date: Tue, 7 Jan 2025 14:49:23 +0100 Subject: [PATCH 16/31] Address #344 by showing blocks when they are necessary (#762) --- .../scala/effekt/core/PrettyPrinter.scala | 36 +++++++++++-------- 1 file changed, 22 insertions(+), 14 deletions(-) diff --git a/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala b/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala index 5a06a46e5..5d79369fd 100644 --- a/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala +++ b/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala @@ -80,7 +80,7 @@ object PrettyPrinter extends ParenPrettyPrinter { def toDoc(b: Block): Doc = b match { case BlockVar(id, _, _) => toDoc(id) case BlockLit(tps, cps, vps, bps, body) => - braces { space <> paramsToDoc(tps, vps, bps) <+> "=>" <+> nest(line <> toDoc(body)) <> line } + braces { space <> paramsToDoc(tps, vps, bps) <+> "=>" <+> nest(line <> toDocStmts(body)) <> line } case Unbox(e) => parens("unbox" <+> toDoc(e)) case New(handler) => "new" <+> toDoc(handler) } @@ -153,7 +153,7 @@ object PrettyPrinter extends ParenPrettyPrinter { def toDoc(d: Toplevel): Doc = d match { case Toplevel.Def(id, BlockLit(tps, cps, vps, bps, body)) => - "def" <+> toDoc(id) <> paramsToDoc(tps, vps, bps) <+> "=" <+> block(toDoc(body)) + "def" <+> toDoc(id) <> paramsToDoc(tps, vps, bps) <+> "=" <+> toDoc(body) case Toplevel.Def(id, block) => "def" <+> toDoc(id) <+> "=" <+> toDoc(block) case Toplevel.Val(id, _, binding) => @@ -161,28 +161,34 @@ object PrettyPrinter extends ParenPrettyPrinter { } def toDoc(s: Stmt): Doc = s match { + // requires a block to be readable: + case _ : (Stmt.Def | Stmt.Let | Stmt.Val | Stmt.Alloc | Stmt.Var) => block(toDocStmts(s)) + case other => toDocStmts(s) + } + + def toDocStmts(s: Stmt): Doc = s match { case Def(id, BlockLit(tps, cps, vps, bps, body), rest) => - "def" <+> toDoc(id) <> paramsToDoc(tps, vps, bps) <+> "=" <+> block(toDoc(body)) <> line <> - toDoc(rest) + "def" <+> toDoc(id) <> paramsToDoc(tps, vps, bps) <+> "=" <+> toDoc(body) <> line <> + toDocStmts(rest) case Def(id, block, rest) => "def" <+> toDoc(id) <+> "=" <+> toDoc(block) <> line <> - toDoc(rest) + toDocStmts(rest) case Let(id, _, binding, rest) => "let" <+> toDoc(id) <+> "=" <+> toDoc(binding) <> line <> - toDoc(rest) + toDocStmts(rest) case Return(e) => "return" <+> toDoc(e) case Val(Wildcard(), _, binding, body) => toDoc(binding) <> ";" <> line <> - toDoc(body) + toDocStmts(body) case Val(id, tpe, binding, body) => - "val" <+> toDoc(id) <> ":" <+> toDoc(tpe) <+> "=" <+> block(toDoc(binding)) <> ";" <> line <> - toDoc(body) + "val" <+> toDoc(id) <> ":" <+> toDoc(tpe) <+> "=" <+> toDoc(binding) <> ";" <> line <> + toDocStmts(body) case App(b, targs, vargs, bargs) => toDoc(b) <> argsToDoc(targs, vargs, bargs) @@ -191,7 +197,7 @@ object PrettyPrinter extends ParenPrettyPrinter { toDoc(b) <> "." <> method.name.toString <> argsToDoc(targs, vargs, bargs) case If(cond, thn, els) => - "if" <+> parens(toDoc(cond)) <+> block(toDoc(thn)) <+> "else" <+> block(toDoc(els)) + "if" <+> parens(toDoc(cond)) <+> block(toDocStmts(thn)) <+> "else" <+> block(toDocStmts(els)) case Reset(body) => "reset" <+> toDoc(body) @@ -200,13 +206,15 @@ object PrettyPrinter extends ParenPrettyPrinter { "shift" <> parens(toDoc(prompt)) <+> toDoc(body) case Resume(k, body) => - "resume" <> parens(toDoc(k)) <+> block(toDoc(body)) + "resume" <> parens(toDoc(k)) <+> block(toDocStmts(body)) case Alloc(id, init, region, body) => - "var" <+> toDoc(id) <+> "in" <+> toDoc(region) <+> "=" <+> toDoc(init) <+> ";" <> line <> toDoc(body) + "var" <+> toDoc(id) <+> "in" <+> toDoc(region) <+> "=" <+> toDoc(init) <+> ";" <> line <> + toDocStmts(body) case Var(id, init, cap, body) => - "var" <+> toDoc(id) <+> "=" <+> toDoc(init) <+> ";" <> line <> toDoc(body) + "var" <+> toDoc(id) <+> "=" <+> toDoc(init) <+> ";" <> line <> + toDocStmts(body) case Get(id, capt, tpe) => "!" <> toDoc(id) @@ -219,7 +227,7 @@ object PrettyPrinter extends ParenPrettyPrinter { case Match(sc, clauses, default) => val cs = braces(nest(line <> vsep(clauses map { case (p, b) => "case" <+> toDoc(p) <+> toDoc(b) })) <> line) - val d = default.map { body => space <> "else" <+> braces(nest(line <> toDoc(body))) }.getOrElse { emptyDoc } + val d = default.map { body => space <> "else" <+> braces(nest(line <> toDocStmts(body))) }.getOrElse { emptyDoc } toDoc(sc) <+> "match" <+> cs <> d case Hole() => From b179d1f8ae168216eb01ef12e5c6381d18edf581 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jonathan=20Immanuel=20Brachth=C3=A4user?= Date: Tue, 7 Jan 2025 14:55:50 +0100 Subject: [PATCH 17/31] Improve dead code elimination to remove unused cases and operations (#763) --- .../effekt/core/optimizer/Deadcode.scala | 55 +++++++++++++++---- 1 file changed, 44 insertions(+), 11 deletions(-) diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/Deadcode.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/Deadcode.scala index a9934a286..776c2c011 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/Deadcode.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/Deadcode.scala @@ -4,26 +4,59 @@ package optimizer class Deadcode(reachable: Map[Id, Usage]) extends core.Tree.Rewrite { + private def used(id: Id): Boolean = reachable.get(id).exists(u => u != Usage.Never) + + private def unused(id: Id): Boolean = !used(id) + override def stmt = { // Remove local unused definitions - case Stmt.Def(id, block, body) if !reachable.isDefinedAt(id) => rewrite(body) - case Stmt.Let(id, tpe, binding, body) if binding.capt.isEmpty && !reachable.isDefinedAt(id) => rewrite(body) + case Stmt.Def(id, block, body) if unused(id) => rewrite(body) + case Stmt.Let(id, tpe, binding, body) if binding.capt.isEmpty && unused(id) => rewrite(body) case Stmt.Reset(body) => rewrite(body) match { - case BlockLit(tparams, cparams, vparams, List(prompt), body) if !reachable.isDefinedAt(prompt.id) => body + case BlockLit(tparams, cparams, vparams, List(prompt), body) if unused(prompt.id) => body case b => Stmt.Reset(b) } + + case Stmt.Match(sc, clauses, default) => + Stmt.Match(rewrite(sc), clauses.collect { + case (id, clause) if used(id) => (id, rewrite(clause)) + }, default.map(rewrite)) + } + + override def implementation = { + case Implementation(interface, operations) => + Implementation(rewrite(interface), operations.collect { + case op if used(op.name) => rewrite(op) + }) + } + + override def rewrite(m: ModuleDecl): ModuleDecl = m match { + case ModuleDecl(path, includes, declarations, externs, definitions, exports) => + ModuleDecl(path, includes, + // drop unused constructors and operations + declarations.map(rewrite), + // drop unreachable externs + m.externs.collect { + case e: Extern.Def if used(e.id) => e + case e: Extern.Include => e + }, + // drop unreachable definitions + definitions.collect { case d if used(d.id) => rewrite(d) }, + exports) } - override def rewrite(m: ModuleDecl): ModuleDecl = m.copy( - // Remove top-level unused definitions - definitions = m.definitions.collect { case d if reachable.isDefinedAt(d.id) => rewrite(d) }, - externs = m.externs.collect { - case e: Extern.Def if reachable.isDefinedAt(e.id) => e - case e: Extern.Include => e - } - ) + def rewrite(d: Declaration): Declaration = d match { + case Declaration.Data(id, tparams, constructors) => + Declaration.Data(id, tparams, constructors.collect { + case c if used(c.id) => c + }) + case Declaration.Interface(id, tparams, properties) => + Declaration.Interface(id, tparams, properties.collect { + case p if used(p.id) => p + }) + } } object Deadcode { From 4a71622ce3cf481d84992ed4563a8d2c7d34af1d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jonathan=20Immanuel=20Brachth=C3=A4user?= Date: Tue, 7 Jan 2025 23:08:18 +0100 Subject: [PATCH 18/31] Add first draft of instrumented virtual machine (#713) In this PR I develop a tracing virtual machine for Core. This will allow us to: - check whether optimizations actually optimize things - measure different aspects of the evaluation as part of CI and make sure that changes do not invalidate optimizations - implement a debugger / visualizer in the long run --- .../src/test/scala/effekt/EffektTests.scala | 20 +- .../src/test/scala/effekt/core/VMTests.scala | 746 ++++++++++++++++++ .../src/main/scala/effekt/core/Parser.scala | 13 +- .../effekt/core/PolymorphismBoxing.scala | 1 - .../effekt/core/optimizer/Normalizer.scala | 1 - .../effekt/core/optimizer/Reachable.scala | 2 + .../main/scala/effekt/core/vm/Builtin.scala | 270 +++++++ .../effekt/core/vm/Instrumentation.scala | 82 ++ .../src/main/scala/effekt/core/vm/VM.scala | 567 +++++++++++++ .../main/scala/effekt/generator/vm/VM.scala | 44 ++ .../benchmarks/are_we_fast_yet/sieve.effekt | 1 - libraries/common/args.effekt | 1 + libraries/common/array.effekt | 4 + libraries/common/effekt.effekt | 56 +- libraries/common/ref.effekt | 3 + 15 files changed, 1788 insertions(+), 23 deletions(-) create mode 100644 effekt/jvm/src/test/scala/effekt/core/VMTests.scala create mode 100644 effekt/shared/src/main/scala/effekt/core/vm/Builtin.scala create mode 100644 effekt/shared/src/main/scala/effekt/core/vm/Instrumentation.scala create mode 100644 effekt/shared/src/main/scala/effekt/core/vm/VM.scala create mode 100644 effekt/shared/src/main/scala/effekt/generator/vm/VM.scala diff --git a/effekt/jvm/src/test/scala/effekt/EffektTests.scala b/effekt/jvm/src/test/scala/effekt/EffektTests.scala index d8de0ba1f..ee3eabbe1 100644 --- a/effekt/jvm/src/test/scala/effekt/EffektTests.scala +++ b/effekt/jvm/src/test/scala/effekt/EffektTests.scala @@ -163,19 +163,19 @@ trait EffektTests extends munit.FunSuite { case f if f.isDirectory && !ignored.contains(f) => f.listFiles.foreach(foreachFileIn(_)(test)) case f if f.getName.endsWith(".effekt") || f.getName.endsWith(".effekt.md") => - val path = f.getParentFile - val baseName = f.getName.stripSuffix(".md").stripSuffix(".effekt") - - if (ignored.contains(f)) { - ignored.contains(f) - } else { - val checkfile = path / (baseName + ".check") - val expected = if checkfile.exists() then Some(IO.read(checkfile)) else None - - test(f, expected) + if (!ignored.contains(f)) { + test(f, expectedResultFor(f)) } case _ => () } runTests() } + +def expectedResultFor(f: File): Option[String] = { + val path = f.getParentFile + val baseName = f.getName.stripSuffix(".md").stripSuffix(".effekt") + val checkfile = path / (baseName + ".check") + if checkfile.exists() then Some(IO.read(checkfile)) else None +} + diff --git a/effekt/jvm/src/test/scala/effekt/core/VMTests.scala b/effekt/jvm/src/test/scala/effekt/core/VMTests.scala new file mode 100644 index 000000000..b7c316ddb --- /dev/null +++ b/effekt/jvm/src/test/scala/effekt/core/VMTests.scala @@ -0,0 +1,746 @@ +package effekt +package core + +import core.vm.* +import effekt.util.MarkdownSource + +class VMTests extends munit.FunSuite { + + import effekt.context.{ Context, IOModuleDB } + import effekt.util.PlainMessaging + import effekt.generator.vm.VM + import kiama.util.{ Positions, StringSource, FileSource } + + + val positions = new Positions + object plainMessaging extends PlainMessaging + object context extends Context(positions) with IOModuleDB { + val messaging = plainMessaging + object frontend extends generator.vm.VM + override lazy val compiler = frontend.asInstanceOf + } + + def compileString(content: String): (Id, symbols.Module, ModuleDecl) = + val config = new EffektConfig(Seq("--Koutput", "string")) + config.verify() + context.setup(config) + context.frontend.compile(StringSource(content, "input.effekt"))(using context).map { + case (_, decl) => decl + }.getOrElse { + val errors = plainMessaging.formatMessages(context.messaging.buffer) + sys error errors + } + + def compileFile(path: String): (Id, symbols.Module, ModuleDecl) = + val config = new EffektConfig(Seq("--Koutput", "string")) + config.verify() + context.setup(config) + + val source = if path.endsWith(".effekt.md") then + MarkdownSource(FileSource(path)) + else + FileSource(path) + + context.frontend.compile(source)(using context).map { + case (_, decl) => decl + }.getOrElse { + val errors = plainMessaging.formatMessages(context.messaging.buffer) + sys error s"Cannot compile ${path}\n\n${errors}" + } + + class OutputCapturingRuntime extends Runtime { + + import java.io.{ByteArrayOutputStream, PrintStream} + + private val outputStream = new ByteArrayOutputStream() + val out = new PrintStream(outputStream) + + def output(): String = { + out.flush() + outputStream.toString + } + } + + def runCounting(main: Id, decl: ModuleDecl): (String, Summary) = + object runtime extends OutputCapturingRuntime + object counting extends Counting + Interpreter(counting, runtime).run(main, decl) + (runtime.output(), Summary( + staticDispatches = counting.staticDispatches, + dynamicDispatches = counting.dynamicDispatches, + patternMatches = counting.patternMatches, + branches = counting.branches, + pushedFrames = counting.pushedFrames, + poppedFrames = counting.poppedFrames, + allocations = counting.allocations, + closures = counting.closures, + fieldLookups = counting.fieldLookups, + variableReads = counting.variableReads, + variableWrites = counting.variableWrites, + resets = counting.resets, + shifts = counting.shifts, + resumes = counting.resumes + )) + + def runString(contents: String): (String, Summary) = + val (main, mod, decl) = compileString(contents) + runCounting(main, decl) + + def runFile(file: String): (String, Summary) = + val (main, mod, decl) = compileFile(file) + runCounting(main, decl) + + + case class Summary( + staticDispatches: Int, + dynamicDispatches: Int, + patternMatches: Int, + branches: Int, + pushedFrames: Int, + poppedFrames: Int, + allocations: Int, + closures: Int, + fieldLookups: Int, + variableReads: Int, + variableWrites: Int, + resets: Int, + shifts: Int, + resumes: Int + ) + + val recursion = + """def fib(n: Int): Int = + | if (n == 0) 1 + | else if (n == 1) 1 + | else fib(n - 2) + fib(n - 1) + | + |def main() = { + | println(fib(10).show) + |} + |""".stripMargin + + + test ("simple recursion") { + assertEquals(runString(recursion)._1, "89\n") + } + + val dynamicDispatch = + """def size[T](l: List[T]): Int = + | l match { + | case Nil() => 0 + | case Cons(hd, tl) => 1 + size(tl) + | } + | + |def map[A, B](l: List[A]) { f: A => B }: List[B] = + | l match { + | case Nil() => Nil() + | case Cons(hd, tl) => Cons(f(hd), map(tl){f}) + | } + | + |def main() = { + | println(size([1, 2, 3].map { x => x + 1 })) + |} + |""".stripMargin + + test ("dynamic dispatch") { + assertEquals(runString(dynamicDispatch)._1, "3\n") + } + + val simpleObject = + """interface Counter { + | def inc(): Unit + | def get(): Int + |} + | + |def main() = { + | def c = new Counter { + | def inc() = println("tick") + | def get() = 0 + | }; + | c.inc() + | c.inc() + | c.inc() + |} + | + |""".stripMargin + + test ("simple object") { + assertEquals(runString(simpleObject)._1, "tick\ntick\ntick\n") + } + + val mutableState = + """def main() = { + | var x = 42; + | x = x + 1; + | println(x.show) + | + | region r { + | var x in r = 10; + | x = x + 1 + | println(x.show) + | } + |} + |""".stripMargin + + test ("mutable state") { + assertEquals(runString(mutableState)._1, "43\n11\n") + } + + val simpleException = + """effect raise(): Unit + | + |def main() = { + | try { + | println("before"); + | do raise() + | println("after") + | } with raise { println("caught") } + |} + | + |""".stripMargin + + test ("simple exception") { + assertEquals(runString(simpleException)._1, "before\ncaught\n") + } + + val sorting = + """import list + | + |def main() = { + | // synchronized with doctest in `sortBy` + | println([1, 3, -1, 5].sortBy { (a, b) => a <= b }) + |} + |""".stripMargin + + test ("sorting (integration)") { + assertEquals(runString(sorting)._1, "Cons(-1, Cons(1, Cons(3, Cons(5, Nil()))))\n") + } + + val toplevelVal = + """ + |val top: Int = 43 + | + |def main() = { + | val x = top + top + | println(x.show) + |} + |""".stripMargin + + test ("toplevel val") { + assertEquals(runString(toplevelVal)._1, "86\n") + } + + + import java.io.File + import sbt.io.* + import sbt.io.syntax.* + + def examplesDir = new File("examples") + + val are_we_fast_yet: Seq[(File, Option[Summary])] = Seq( + examplesDir / "benchmarks" / "are_we_fast_yet" / "bounce.effekt" -> Some(Summary( + staticDispatches = 5202, + dynamicDispatches = 5000, + patternMatches = 0, + branches = 31628, + pushedFrames = 34010, + poppedFrames = 34010, + allocations = 0, + closures = 100, + fieldLookups = 0, + variableReads = 7132, + variableWrites = 3157, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "are_we_fast_yet" / "list_tail.effekt" -> Some(Summary( + staticDispatches = 23713, + dynamicDispatches = 0, + patternMatches = 41728, + branches = 2843, + pushedFrames = 4961, + poppedFrames = 4961, + allocations = 34, + closures = 0, + fieldLookups = 0, + variableReads = 0, + variableWrites = 0, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "are_we_fast_yet" / "mandelbrot.effekt" -> Some(Summary( + staticDispatches = 11, + dynamicDispatches = 0, + patternMatches = 0, + branches = 26, + pushedFrames = 117, + poppedFrames = 117, + allocations = 0, + closures = 0, + fieldLookups = 0, + variableReads = 70, + variableWrites = 32, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "are_we_fast_yet" / "nbody.effekt" -> Some(Summary( + staticDispatches = 56, + dynamicDispatches = 0, + patternMatches = 0, + branches = 56, + pushedFrames = 80, + poppedFrames = 80, + allocations = 31, + closures = 0, + fieldLookups = 455, + variableReads = 34, + variableWrites = 30, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "are_we_fast_yet" / "queens.effekt" -> Some(Summary( + staticDispatches = 1146, + dynamicDispatches = 0, + patternMatches = 0, + branches = 3887, + pushedFrames = 3084, + poppedFrames = 3060, + allocations = 0, + closures = 0, + fieldLookups = 0, + variableReads = 0, + variableWrites = 0, + resets = 113, + shifts = 8, + resumes = 0 + )), + + examplesDir / "benchmarks" / "are_we_fast_yet" / "sieve.effekt" -> Some(Summary( + staticDispatches = 21738, + dynamicDispatches = 0, + patternMatches = 0, + branches = 26736, + pushedFrames = 51284, + poppedFrames = 51284, + allocations = 0, + closures = 0, + fieldLookups = 0, + variableReads = 34546, + variableWrites = 11738, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "are_we_fast_yet" / "towers.effekt" -> Some(Summary( + staticDispatches = 16401, + dynamicDispatches = 0, + patternMatches = 16396, + branches = 16287, + pushedFrames = 65630, + poppedFrames = 65630, + allocations = 8206, + closures = 0, + fieldLookups = 0, + variableReads = 41027, + variableWrites = 8205, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "are_we_fast_yet" / "permute.effekt" -> Some(Summary( + staticDispatches = 17326, + dynamicDispatches = 0, + patternMatches = 0, + branches = 17326, + pushedFrames = 54797, + poppedFrames = 54797, + allocations = 0, + closures = 0, + fieldLookups = 0, + variableReads = 32437, + variableWrites = 13699, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "are_we_fast_yet" / "storage.effekt" -> Some(Summary( + staticDispatches = 5463, + dynamicDispatches = 0, + patternMatches = 0, + branches = 5463, + pushedFrames = 28674, + poppedFrames = 28674, + allocations = 5461, + closures = 0, + fieldLookups = 0, + variableReads = 13654, + variableWrites = 9557, + resets = 0, + shifts = 0, + resumes = 0 + )), + ) + + val duality_of_compilation: Seq[(File, Option[Summary])] = Seq( + examplesDir / "benchmarks" / "duality_of_compilation" / "erase_unused.effekt" -> Some(Summary( + staticDispatches = 21, + dynamicDispatches = 0, + patternMatches = 0, + branches = 21, + pushedFrames = 6, + poppedFrames = 6, + allocations = 16, + closures = 0, + fieldLookups = 0, + variableReads = 0, + variableWrites = 0, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "duality_of_compilation" / "factorial_accumulator.effekt" -> Some(Summary( + staticDispatches = 6, + dynamicDispatches = 0, + patternMatches = 0, + branches = 6, + pushedFrames = 1, + poppedFrames = 1, + allocations = 0, + closures = 0, + fieldLookups = 0, + variableReads = 0, + variableWrites = 0, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "duality_of_compilation" / "fibonacci_recursive.effekt" -> Some(Summary( + staticDispatches = 15, + dynamicDispatches = 0, + patternMatches = 0, + branches = 27, + pushedFrames = 15, + poppedFrames = 15, + allocations = 0, + closures = 0, + fieldLookups = 0, + variableReads = 0, + variableWrites = 0, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "duality_of_compilation" / "iterate_increment.effekt" -> Some(Summary( + staticDispatches = 6, + dynamicDispatches = 0, + patternMatches = 0, + branches = 6, + pushedFrames = 1, + poppedFrames = 1, + allocations = 0, + closures = 0, + fieldLookups = 0, + variableReads = 0, + variableWrites = 0, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "duality_of_compilation" / "lookup_tree.effekt" -> Some(Summary( + staticDispatches = 12, + dynamicDispatches = 0, + patternMatches = 6, + branches = 6, + pushedFrames = 7, + poppedFrames = 7, + allocations = 6, + closures = 0, + fieldLookups = 0, + variableReads = 0, + variableWrites = 0, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "duality_of_compilation" / "match_options.effekt" -> Some(Summary( + staticDispatches = 6, + dynamicDispatches = 0, + patternMatches = 6, + branches = 6, + pushedFrames = 7, + poppedFrames = 7, + allocations = 6, + closures = 0, + fieldLookups = 0, + variableReads = 0, + variableWrites = 0, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "duality_of_compilation" / "sum_range.effekt" -> Some(Summary( + staticDispatches = 12, + dynamicDispatches = 0, + patternMatches = 6, + branches = 6, + pushedFrames = 12, + poppedFrames = 12, + allocations = 6, + closures = 0, + fieldLookups = 0, + variableReads = 0, + variableWrites = 0, + resets = 0, + shifts = 0, + resumes = 0 + )), + ) + + val effect_handlers_bench: Seq[(File, Option[Summary])] = Seq( + examplesDir / "benchmarks" / "effect_handlers_bench" / "countdown.effekt" -> Some(Summary( + staticDispatches = 6, + dynamicDispatches = 0, + patternMatches = 0, + branches = 6, + pushedFrames = 12, + poppedFrames = 12, + allocations = 0, + closures = 0, + fieldLookups = 0, + variableReads = 6, + variableWrites = 5, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "effect_handlers_bench" / "iterator.effekt" -> Some(Summary( + staticDispatches = 7, + dynamicDispatches = 0, + patternMatches = 0, + branches = 7, + pushedFrames = 14, + poppedFrames = 14, + allocations = 0, + closures = 0, + fieldLookups = 0, + variableReads = 7, + variableWrites = 6, + resets = 0, + shifts = 0, + resumes = 0 + )), + + examplesDir / "benchmarks" / "effect_handlers_bench" / "nqueens.effekt" -> Some(Summary( + staticDispatches = 626, + dynamicDispatches = 0, + patternMatches = 400, + branches = 1487, + pushedFrames = 1352, + poppedFrames = 1409, + allocations = 54, + closures = 0, + fieldLookups = 0, + variableReads = 0, + variableWrites = 0, + resets = 1, + shifts = 211, + resumes = 220 + )), + + examplesDir / "benchmarks" / "effect_handlers_bench" / "parsing_dollars.effekt" -> Some(Summary( + staticDispatches = 67, + dynamicDispatches = 0, + patternMatches = 0, + branches = 210, + pushedFrames = 379, + poppedFrames = 377, + allocations = 0, + closures = 0, + fieldLookups = 0, + variableReads = 222, + variableWrites = 88, + resets = 1, + shifts = 1, + resumes = 0 + )), + + examplesDir / "benchmarks" / "effect_handlers_bench" / "product_early.effekt" -> Some(Summary( + staticDispatches = 6013, + dynamicDispatches = 0, + patternMatches = 5005, + branches = 6013, + pushedFrames = 6008, + poppedFrames = 1008, + allocations = 1002, + closures = 0, + fieldLookups = 0, + variableReads = 0, + variableWrites = 0, + resets = 5, + shifts = 5, + resumes = 0 + )), + + examplesDir / "benchmarks" / "effect_handlers_bench" / "resume_nontail.effekt" -> Some(Summary( + staticDispatches = 7001, + dynamicDispatches = 0, + patternMatches = 0, + branches = 12001, + pushedFrames = 16001, + poppedFrames = 16001, + allocations = 0, + closures = 0, + fieldLookups = 0, + variableReads = 0, + variableWrites = 0, + resets = 1000, + shifts = 5000, + resumes = 5000 + )), + + examplesDir / "benchmarks" / "effect_handlers_bench" / "tree_explore.effekt" -> Some(Summary( + staticDispatches = 3187, + dynamicDispatches = 0, + patternMatches = 3490, + branches = 3167, + pushedFrames = 8207, + poppedFrames = 9807, + allocations = 2556, + closures = 0, + fieldLookups = 0, + variableReads = 2051, + variableWrites = 1430, + resets = 10, + shifts = 310, + resumes = 620 + )), + + examplesDir / "benchmarks" / "effect_handlers_bench" / "triples.effekt" -> Some(Summary( + staticDispatches = 231, + dynamicDispatches = 0, + patternMatches = 4, + branches = 701, + pushedFrames = 874, + poppedFrames = 880, + allocations = 4, + closures = 0, + fieldLookups = 0, + variableReads = 0, + variableWrites = 0, + resets = 1, + shifts = 347, + resumes = 350 + )), + ) + + val casestudies: Seq[(File, Option[Summary])] = Seq( + examplesDir / "casestudies" / "ad.effekt.md" -> Some(Summary( + staticDispatches = 19, + dynamicDispatches = 335, + patternMatches = 0, + branches = 285, + pushedFrames = 453, + poppedFrames = 453, + allocations = 174, + closures = 39, + fieldLookups = 706, + variableReads = 0, + variableWrites = 0, + resets = 8, + shifts = 29, + resumes = 29 + )), + + examplesDir / "casestudies" / "buildsystem.effekt.md" -> Some(Summary( + staticDispatches = 43, + dynamicDispatches = 57, + patternMatches = 31, + branches = 40, + pushedFrames = 33, + poppedFrames = 30, + allocations = 15, + closures = 36, + fieldLookups = 0, + variableReads = 7, + variableWrites = 3, + resets = 9, + shifts = 7, + resumes = 4 + )), + + examplesDir / "casestudies" / "scheduler.effekt.md" -> Some(Summary( + staticDispatches = 60, + dynamicDispatches = 8, + patternMatches = 95, + branches = 41, + pushedFrames = 106, + poppedFrames = 106, + allocations = 73, + closures = 8, + fieldLookups = 0, + variableReads = 29, + variableWrites = 18, + resets = 1, + shifts = 7, + resumes = 7 + )), + + examplesDir / "casestudies" / "inference.effekt.md" -> Some(Summary( + staticDispatches = 1457444, + dynamicDispatches = 3201452, + patternMatches = 1474376, + branches = 303298, + pushedFrames = 7574572, + poppedFrames = 6709277, + allocations = 4626007, + closures = 865541, + fieldLookups = 0, + variableReads = 2908620, + variableWrites = 1453663, + resets = 288559, + shifts = 297723, + resumes = 9275 + )), + ) + + val testFiles: Seq[(File, Option[Summary])] = + are_we_fast_yet ++ + duality_of_compilation ++ + effect_handlers_bench ++ + casestudies + + def runTest(f: File, expectedSummary: Option[Summary]): Unit = + val path = f.getPath + test(path) { + try { + val (result, summary) = runFile(path) + val expected = expectedResultFor(f).getOrElse { s"Missing checkfile for ${path}"} + assertNoDiff(result, expected) + expectedSummary.foreach { expected => assertEquals(summary, expected) } + } catch { + case i: VMError => fail(i.getMessage, i) + } + } + + testFiles.foreach(runTest) + + +} diff --git a/effekt/shared/src/main/scala/effekt/core/Parser.scala b/effekt/shared/src/main/scala/effekt/core/Parser.scala index b2175012a..1c77732e0 100644 --- a/effekt/shared/src/main/scala/effekt/core/Parser.scala +++ b/effekt/shared/src/main/scala/effekt/core/Parser.scala @@ -176,7 +176,7 @@ class CoreParsers(positions: Positions, names: Names) extends EffektLexers(posit | id ~ (`:` ~> valueType) ^^ Pure.ValueVar.apply | `box` ~> captures ~ block ^^ { case capt ~ block => Pure.Box(block, capt) } | `make` ~> dataType ~ id ~ valueArgs ^^ Pure.Make.apply - | maybeParensBlockVar ~ maybeTypeArgs ~ valueArgs ^^ Pure.PureApp.apply + | maybeParens(blockVar) ~ maybeTypeArgs ~ valueArgs ^^ Pure.PureApp.apply | failure("Expected a pure expression.") ) @@ -196,9 +196,11 @@ class CoreParsers(positions: Positions, names: Names) extends EffektLexers(posit // ----------- lazy val expr: P[Expr] = ( pure - | (`!` ~/> maybeParensBlockVar) ~ maybeTypeArgs ~ valueArgs ~ blockArgs ^^ DirectApp.apply + | (`!` ~/> maybeParens(blockVar)) ~ maybeTypeArgs ~ valueArgs ~ blockArgs ^^ DirectApp.apply ) + def maybeParens[T](p: P[T]): P[T] = (p | `(` ~> p <~ `)`) + // Blocks // ------ @@ -213,14 +215,9 @@ class CoreParsers(positions: Positions, names: Names) extends EffektLexers(posit lazy val blockVar: P[Block.BlockVar] = id ~ (`:` ~> blockType) ~ (`@` ~> captures) ^^ { - case (id ~ tpe ~ capt) => Block.BlockVar(id, tpe, capt) : Block.BlockVar + case id ~ tpe ~ capt => Block.BlockVar(id, tpe, capt) : Block.BlockVar } - lazy val maybeParensBlockVar: P[Block.BlockVar] = - ( `(` ~> blockVar <~ `)` - | blockVar - ) - lazy val blockLit: P[Block.BlockLit] = `{` ~> parameters ~ (`=>` ~/> stmts) <~ `}` ^^ { case (tparams, cparams, vparams, bparams) ~ body => diff --git a/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala b/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala index 553f4d7c8..ee052bf1e 100644 --- a/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala +++ b/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala @@ -524,7 +524,6 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { def isIdentity: Boolean } object BlockCoercer { - def apply(from: BlockType, to: BlockType, targs: List[ValueType] = Nil)(using PContext): BlockCoercer = (from, to) match { case (f, t) if f == t => IdentityCoercer(f, t) diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala index e6a548ed0..02f23c3c6 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala @@ -310,7 +310,6 @@ object Normalizer { normal => def normalize(e: Expr)(using Context): Expr = e match { case DirectApp(b, targs, vargs, bargs) => DirectApp(b, targs, vargs.map(normalize), bargs.map(normalize)) - case pure: Pure => normalize(pure) } diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala index 5ea5e008e..dc5aab2f6 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala @@ -4,6 +4,8 @@ package optimizer /** * A simple reachability analysis. + * + * TODO reachability should also process externs since they now contain splices. */ class Reachable( var reachable: Map[Id, Usage], diff --git a/effekt/shared/src/main/scala/effekt/core/vm/Builtin.scala b/effekt/shared/src/main/scala/effekt/core/vm/Builtin.scala new file mode 100644 index 000000000..0953259ce --- /dev/null +++ b/effekt/shared/src/main/scala/effekt/core/vm/Builtin.scala @@ -0,0 +1,270 @@ +package effekt +package core +package vm + +import java.io.PrintStream + +trait Runtime { + def out: PrintStream +} +def Runtime(using run: Runtime) = run + +case class Builtin(name: String, impl: Runtime => List[Value] ~> Value) + +def builtin(name: String)(impl: Runtime ?=> List[Value] ~> Value): (String, Builtin) = + name -> Builtin(name, env => impl(using env)) + +type Builtins = Map[String, Builtin] + +lazy val printing: Builtins = Map( + builtin("effekt::println(String)") { + case As.String(msg) :: Nil => + Runtime.out.println(msg) + Value.Unit() + }, +) + +lazy val doubles: Builtins = Map( + // Arithmetic + // ---------- + builtin("effekt::infixAdd(Double, Double)") { + case As.Double(x) :: As.Double(y) :: Nil => Value.Double(x + y) + }, + builtin("effekt::infixSub(Double, Double)") { + case As.Double(x) :: As.Double(y) :: Nil => Value.Double(x - y) + }, + builtin("effekt::infixMul(Double, Double)") { + case As.Double(x) :: As.Double(y) :: Nil => Value.Double(x * y) + }, + builtin("effekt::infixDiv(Double, Double)") { + case As.Double(x) :: As.Double(y) :: Nil => Value.Double(x / y) + }, + builtin("effekt::sqrt(Double)") { + case As.Double(x) :: Nil => Value.Double(Math.sqrt(x)) + }, + builtin("effekt::exp(Double)") { + case As.Double(x) :: Nil => Value.Double(Math.exp(x)) + }, + builtin("effekt::log(Double)") { + case As.Double(x) :: Nil => Value.Double(Math.log(x)) + }, + builtin("effekt::cos(Double)") { + case As.Double(x) :: Nil => Value.Double(Math.cos(x)) + }, + builtin("effekt::sin(Double)") { + case As.Double(x) :: Nil => Value.Double(Math.sin(x)) + }, + builtin("effekt::tan(Double)") { + case As.Double(x) :: Nil => Value.Double(Math.tan(x)) + }, + builtin("effekt::atan(Double)") { + case As.Double(x) :: Nil => Value.Double(Math.atan(x)) + }, + builtin("effekt::round(Double)") { + case As.Double(x) :: Nil => Value.Int(Math.round(x)) + }, + builtin("effekt::pow(Double, Double)") { + case As.Double(base) :: As.Double(exp) :: Nil => Value.Double(Math.pow(base, exp)) + }, + builtin("effekt::pi()") { + case Nil => Value.Double(Math.PI) + }, + + + // Comparison + // ---------- + builtin("effekt::infixEq(Double, Double)") { + case As.Double(x) :: As.Double(y) :: Nil => Value.Bool(x == y) + }, + builtin("effekt::infixNeq(Double, Double)") { + case As.Double(x) :: As.Double(y) :: Nil => Value.Bool(x != y) + }, + builtin("effekt::infixLt(Double, Double)") { + case As.Double(x) :: As.Double(y) :: Nil => Value.Bool(x < y) + }, + builtin("effekt::infixGt(Double, Double)") { + case As.Double(x) :: As.Double(y) :: Nil => Value.Bool(x > y) + }, + builtin("effekt::infixLte(Double, Double)") { + case As.Double(x) :: As.Double(y) :: Nil => Value.Bool(x <= y) + }, + builtin("effekt::infixGte(Double, Double)") { + case As.Double(x) :: As.Double(y) :: Nil => Value.Bool(x >= y) + }, + + // Conversion + // ---------- + builtin("effekt::toInt(Double)") { + case As.Double(x) :: Nil => Value.Int(x.toLong) + }, + builtin("effekt::show(Double)") { + // TODO globally define show on double in a decent way... this mimicks JS + case As.Double(n) :: Nil => + Value.String( + if (n == n.toInt.toDouble) n.toInt.toString // Handle integers like 15.0 β†’ 15 + else { + val formatted = BigDecimal(n) + .setScale(15, BigDecimal.RoundingMode.DOWN) // Truncate to 15 decimal places without rounding up + .bigDecimal + .stripTrailingZeros() + .toPlainString + + formatted + } + ) + }, +) + +lazy val integers: Builtins = Map( + // Arithmetic + // ---------- + builtin("effekt::infixAdd(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Int(x + y) + }, + builtin("effekt::infixSub(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Int(x - y) + }, + builtin("effekt::infixMul(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Int(x * y) + }, + builtin("effekt::mod(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Int(x % y) + }, + builtin("effekt::infixDiv(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Int(x / y) + }, + builtin("effekt::bitwiseShl(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Int(x << y) + }, + builtin("effekt::bitwiseShr(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Int(x >> y) + }, + builtin("effekt::bitwiseAnd(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Int(x & y) + }, + builtin("effekt::bitwiseOr(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Int(x | y) + }, + builtin("effekt::bitwiseXor(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Int(x ^ y) + }, + + // Comparison + // ---------- + builtin("effekt::infixEq(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Bool(x == y) + }, + builtin("effekt::infixNeq(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Bool(x != y) + }, + builtin("effekt::infixLt(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Bool(x < y) + }, + builtin("effekt::infixGt(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Bool(x > y) + }, + builtin("effekt::infixLte(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Bool(x <= y) + }, + builtin("effekt::infixGte(Int, Int)") { + case As.Int(x) :: As.Int(y) :: Nil => Value.Bool(x >= y) + }, + + // Conversion + // ---------- + builtin("effekt::toDouble(Int)") { + case As.Int(x) :: Nil => Value.Double(x.toDouble) + }, + + builtin("effekt::show(Int)") { + case As.Int(n) :: Nil => Value.String(n.toString) + }, +) + +lazy val booleans: Builtins = Map( + builtin("effekt::not(Bool)") { + case As.Bool(x) :: Nil => Value.Bool(!x) + }, +) + +lazy val strings: Builtins = Map( + builtin("effekt::infixConcat(String, String)") { + case As.String(x) :: As.String(y) :: Nil => Value.String(x + y) + }, + + builtin("effekt::inspect(Any)") { + case any :: Nil => Value.String(inspect(any)) + }, + + builtin("effekt::infixEq(String, String)") { + case As.String(x) :: As.String(y) :: Nil => Value.Bool(x == y) + }, +) + +lazy val arrays: Builtins = Map( + builtin("array::allocate(Int)") { + case As.Int(x) :: Nil => Value.Array(scala.Array.ofDim(x.toInt)) + }, + builtin("array::size[T](Array[T])") { + case As.Array(arr) :: Nil => Value.Int(arr.length.toLong) + }, + builtin("array::unsafeGet[T](Array[T], Int)") { + case As.Array(arr) :: As.Int(index) :: Nil => arr(index.toInt) + }, + builtin("array::unsafeSet[T](Array[T], Int, T)") { + case As.Array(arr) :: As.Int(index) :: value :: Nil => arr.update(index.toInt, value); Value.Unit() + }, +) + +lazy val refs: Builtins = Map( + builtin("ref::ref[T](T)") { + case init :: Nil => Value.Ref(Reference(init)) + }, + builtin("ref::get[T](Ref[T])") { + case As.Reference(ref) :: Nil => ref.value + }, + builtin("ref::set[T](Ref[T], T)") { + case As.Reference(ref) :: value :: Nil => ref.value = value; Value.Unit() + }, +) + +lazy val builtins: Builtins = printing ++ integers ++ doubles ++ booleans ++ strings ++ arrays ++ refs + +protected object As { + object String { + def unapply(v: Value): Option[java.lang.String] = v match { + case Value.Literal(value: java.lang.String) => Some(value) + case _ => None + } + } + object Int { + def unapply(v: Value): Option[scala.Long] = v match { + case Value.Literal(value: scala.Long) => Some(value) + case _ => None + } + } + object Bool { + def unapply(v: Value): Option[scala.Boolean] = v match { + case Value.Literal(value: scala.Boolean) => Some(value) + case _ => None + } + } + object Double { + def unapply(v: Value): Option[scala.Double] = v match { + case Value.Literal(value: scala.Double) => Some(value) + case _ => None + } + } + object Array { + def unapply(v: Value): Option[scala.Array[Value]] = v match { + case Value.Array(array) => Some(array) + case _ => None + } + } + object Reference { + def unapply(v: Value): Option[Reference] = v match { + case Value.Ref(ref) => Some(ref) + case _ => None + } + } +} diff --git a/effekt/shared/src/main/scala/effekt/core/vm/Instrumentation.scala b/effekt/shared/src/main/scala/effekt/core/vm/Instrumentation.scala new file mode 100644 index 000000000..31f89ab3e --- /dev/null +++ b/effekt/shared/src/main/scala/effekt/core/vm/Instrumentation.scala @@ -0,0 +1,82 @@ +package effekt +package core +package vm + +trait Instrumentation { + def staticDispatch(id: Id): Unit = () + def dynamicDispatch(id: Id): Unit = () + def patternMatch(comparisons: Int): Unit = () + def branch(): Unit = () + def pushFrame(): Unit = () + def popFrame(): Unit = () + def allocate(v: Value.Data): Unit = () + def closure(): Unit = () + def fieldLookup(id: Id): Unit = () + def step(state: State): Unit = () + def readMutableVariable(id: Id): Unit = () + def writeMutableVariable(id: Id): Unit = () + def allocateVariable(id: Id): Unit = () + def allocateRegion(region: Address): Unit = () + def allocateVariableIntoRegion(id: Id, region: Address): Unit = () + def reset(): Unit = () + def shift(): Unit = () + def resume(): Unit = () + def builtin(name: String): Unit = () +} +object NoInstrumentation extends Instrumentation + +class Counting extends Instrumentation { + var staticDispatches = 0 + var dynamicDispatches = 0 + var patternMatches = 0 + var branches = 0 + var pushedFrames = 0 + var poppedFrames = 0 + var allocations = 0 + var closures = 0 + var fieldLookups = 0 + var variableReads = 0 + var variableWrites = 0 + + var resets = 0 + var shifts = 0 + var resumes = 0 + + override def staticDispatch(id: Id): Unit = staticDispatches += 1 + override def dynamicDispatch(id: Id): Unit = dynamicDispatches += 1 + override def patternMatch(comparisons: Int): Unit = patternMatches += 1 + override def branch(): Unit = branches += 1 + override def pushFrame(): Unit = pushedFrames += 1 + override def popFrame(): Unit = poppedFrames += 1 + override def allocate(v: Value.Data): Unit = allocations += 1 + override def closure(): Unit = closures += 1 + override def fieldLookup(id: Id): Unit = fieldLookups += 1 + override def readMutableVariable(id: Id): Unit = variableReads += 1 + override def writeMutableVariable(id: Id): Unit = variableWrites += 1 + override def reset(): Unit = resets += 1 + override def shift(): Unit = shifts += 1 + override def resume(): Unit = resumes += 1 + + def report() = + println(s"Static dispatches: ${staticDispatches}") + println(s"Dynamic dispatches: ${dynamicDispatches}") + println(s"Branches: ${branches}") + println(s"Pattern matches: ${patternMatches}") + println(s"Frames (pushed: ${pushedFrames}, popped: ${poppedFrames})") + println(s"Allocations: ${allocations}") + println(s"Closures: ${closures}") + println(s"Field lookups: ${fieldLookups}") + println(s"Variable reads: ${variableReads}") + println(s"Variable writes: ${variableWrites}") + println(s"Installed delimiters: ${resets}") + println(s"Captured continuations: ${shifts}") + println(s"Resumed continuations: ${resumes}") +} + +trait BuiltinHistogram extends Instrumentation { + var builtins: Map[String, Int] = Map.empty + + override def builtin(name: String): Unit = + val before = builtins.getOrElse(name, 0) + builtins = builtins.updated(name, before + 1) +} diff --git a/effekt/shared/src/main/scala/effekt/core/vm/VM.scala b/effekt/shared/src/main/scala/effekt/core/vm/VM.scala new file mode 100644 index 000000000..60e32311b --- /dev/null +++ b/effekt/shared/src/main/scala/effekt/core/vm/VM.scala @@ -0,0 +1,567 @@ +package effekt +package core +package vm + +import effekt.core.vm.Computation.Reference +import effekt.source.FeatureFlag + +import scala.annotation.tailrec + + +type ~>[-A, +B] = PartialFunction[A, B] + + +type Address = Int +private var lastAddress: Address = 0 +def freshAddress(): Address = { lastAddress += 1; lastAddress } + +val GLOBAL_PROMPT = 0 + +class Reference(var value: Value) + +enum Value { + case Literal(value: Any) + // TODO this could also be Pointer(Array | Ref) + case Array(array: scala.Array[Value]) + case Ref(ref: Reference) + case Data(data: ValueType.Data, tag: Id, fields: List[Value]) + case Boxed(block: Computation) +} +object Value { + def Int(v: Long): Value = Value.Literal(v) + def Bool(b: Boolean): Value = Value.Literal(b) + def Unit(): Value = Value.Literal(()) + def Double(d: scala.Double): Value = Value.Literal(d) + def String(s: java.lang.String): Value = Value.Literal(s) +} + +def inspect(v: Value): String = v match { + case Value.Literal(value) => value.toString + case Value.Data(data, tag, fields) => + tag.name.name + "(" + fields.map(inspect).mkString(", ") + ")" + case Value.Boxed(block) => block.toString + case Value.Array(arr) => ??? + case Value.Ref(ref) => ??? +} + +enum Computation { + case Closure(id: Id, env: Env) + case Object(methods: Map[Id, BlockLit], env: Env) + case Region(address: Address) + case Prompt(address: Address) + case Reference(region: Address) + case Resumption(cont: Stack) +} + +enum Env { + case Top(functions: Map[Id, BlockLit], builtins: Map[Id, Builtin], toplevel: Map[Id, Value], declarations: List[core.Declaration]) + case Static(id: Id, block: BlockLit, rest: Env) + case Dynamic(id: Id, block: Computation, rest: Env) + case Let(id: Id, value: Value, rest: Env) + + def bind(id: Id, value: Value): Env = Let(id, value, this) + def bind(id: Id, lit: BlockLit): Env = Static(id, lit, this) + def bind(id: Id, block: Computation): Env = Dynamic(id, block, this) + def bindValues(otherValues: List[(Id, Value)]): Env = + otherValues.foldLeft(this) { case (env, (id, value)) => Let(id, value, env) } + def bindBlocks(otherBlocks: List[(Id, Computation)]): Env = + otherBlocks.foldLeft(this) { case (env, (id, block)) => Dynamic(id, block, env) } + + def lookupValue(id: Id): Value = { + @tailrec + def go(rest: Env): Value = rest match { + case Env.Top(functions, builtins, toplevel, declarations) => + toplevel.getOrElse(id, throw VMError.NotFound(id)) + case Env.Static(id, block, rest) => go(rest) + case Env.Dynamic(id, block, rest) => go(rest) + case Env.Let(otherId, value, rest) => if (id == otherId) value else go(rest) + } + go(this) + } + + def lookupBuiltin(id: Id): Builtin = { + @tailrec + def go(rest: Env): Builtin = rest match { + case Env.Top(functions, builtins, toplevel, declarations) => builtins.getOrElse(id, throw VMError.NotFound(id)) + case Env.Static(id, block, rest) => go(rest) + case Env.Dynamic(id, block, rest) => go(rest) + case Env.Let(id, value, rest) => go(rest) + } + go(this) + } + + def lookupStatic(id: Id): (BlockLit, Env) = this match { + case Env.Top(functions, builtins, toplevel, declarations) => (functions.getOrElse(id, throw VMError.NotFound(id)), this) + case Env.Static(other, block, rest) => if (id == other) (block, this) else rest.lookupStatic(id) + case Env.Dynamic(other, block, rest) => rest.lookupStatic(id) + case Env.Let(other, value, rest) => rest.lookupStatic(id) + } +} + +enum VMError extends Throwable { + case NotFound(id: Id) + case NotAnExternFunction(id: Id) + case MissingBuiltin(name: String) + case RuntimeTypeError(msg: String) + case NonExhaustive(missingCase: Id) + case Hole() + case NoMain() + + override def getMessage: String = this match { + case VMError.NotFound(id) => s"Not found: ${id}" + case VMError.NotAnExternFunction(id) => s"Not an extern function: ${id}" + case VMError.MissingBuiltin(name) => s"Missing builtin: ${name}" + case VMError.RuntimeTypeError(msg) => s"Runtime type error: ${msg}" + case VMError.NonExhaustive(missingCase) => s"Non exhaustive: ${missingCase}" + case VMError.Hole() => s"Reached hole" + case VMError.NoMain() => s"No main" + } +} + +enum Stack { + case Empty + case Segment(frames: List[Frame], prompt: Address, rest: Stack) +} +object Stack { + val Toplevel = Stack.Segment(Nil, GLOBAL_PROMPT, Stack.Empty) +} +def show(stack: Stack): String = stack match { + case Stack.Empty => "Empty" + case Stack.Segment(frames, prompt, rest) => + s"${frames.map(show).mkString(" :: ")} :: p${prompt } :: ${show(rest)}" +} + +def show(frame: Frame): String = frame match { + case Frame.Var(x, value) => s"${util.show(x)}=${show(value)}" + case Frame.Val(x, body, env) => s"val ${util.show(x)}" + case Frame.Region(r, values) => s"region ${r} {${values.map { + case (id, value) => s"${util.show(id)}=${show(value)}}" + }.mkString(", ")}}" +} + +def show(value: Value): String = inspect(value) + +enum Frame { + // mutable state + case Var(x: Address, value: Value) + // sequencing + case Val(x: Id, body: Stmt, env: Env) + // local regions + case Region(r: Address, values: Heap) +} + +type Heap = Map[Address, Value] + +enum State { + case Done(result: Value) + case Step(stmt: Stmt, env: Env, stack: Stack, heap: Heap) +} + +class Interpreter(instrumentation: Instrumentation, runtime: Runtime) { + + // TODO maybe replace region values by integers instead of Id + + @tailrec + private def returnWith(value: Value, env: Env, stack: Stack, heap: Heap): State = + @tailrec + def go(frames: List[Frame], prompt: Address, stack: Stack): State = + frames match { + case Frame.Val(x, body, frameEnv) :: rest => + instrumentation.popFrame() + State.Step(body, frameEnv.bind(x, value), Stack.Segment(rest, prompt, stack), heap) + // free the mutable state + case Frame.Var(x, value) :: rest => go(rest, prompt, stack) + // free the region + case Frame.Region(x, values) :: rest => go(rest, prompt, stack) + case Nil => returnWith(value, env, stack, heap) + } + stack match { + case Stack.Empty => State.Done(value) + case Stack.Segment(frames, prompt, rest) => + go(frames, prompt, rest) + } + + private def push(frame: Frame, stack: Stack): Stack = stack match { + case Stack.Empty => ??? + case Stack.Segment(frames, prompt, rest) => Stack.Segment(frame :: frames, prompt, rest) + } + + @tailrec + private def findFirst[A](stack: Stack)(f: Frame ~> A): Option[A] = + stack match { + case Stack.Empty => None + case Stack.Segment(frames, prompt, rest) => + @tailrec + def go(frames: List[Frame]): Option[A] = + frames match { + case Nil => findFirst(rest)(f) + case frame :: rest if f.isDefinedAt(frame) => Some(f(frame)) + case frame :: rest => go(rest) + } + go(frames) + } + + def updateOnce(stack: Stack)(f: Frame ~> Frame): Stack = + stack match { + case Stack.Empty => ??? + case Stack.Segment(frames, prompt, rest) => + def go(frames: List[Frame], acc: List[Frame]): Stack = + frames match { + case Nil => + Stack.Segment(acc.reverse, prompt, updateOnce(rest)(f)) + case frame :: frames if f.isDefinedAt(frame) => + Stack.Segment(acc.reverse ++ (f(frame) :: frames), prompt, rest) + case frame :: frames => + go(frames, frame :: acc) + } + go(frames, Nil) + } + + @tailrec + private def findFirst[T](env: Env)(f: Env ~> T): T = env match { + case e if f.isDefinedAt(e) => f(e) + case Env.Top(functions, builtins, toplevel, declarations) => ??? + case Env.Static(id, block, rest) => findFirst(rest)(f) + case Env.Dynamic(id, block, rest) => findFirst(rest)(f) + case Env.Let(id, value, rest) => findFirst(rest)(f) + } + + def step(s: State): State = + instrumentation.step(s) + s match { + case State.Done(result) => s + case State.Step(stmt, env, stack, heap) => stmt match { + // do not create a closure + case Stmt.Def(id, block: Block.BlockLit, body) => State.Step(body, env.bind(id, block), stack, heap) + + // create a closure + case Stmt.Def(id, block, body) => State.Step(body, env.bind(id, eval(block, env)), stack, heap) + + case Stmt.Let(id, tpe, binding, body) => State.Step(body, env.bind(id, eval(binding, env)), stack, heap) + + case Stmt.Return(expr) => + val v = eval(expr, env) + returnWith(v, env, stack, heap) + + case Stmt.Val(id, annotatedTpe, binding, body) => + instrumentation.pushFrame() + State.Step(binding, env, push(Frame.Val(id, body, env), stack), heap) + + case Stmt.App(Block.BlockVar(id, _, _), targs, vargs, bargs) => + @tailrec + def lookup(env: Env): (BlockLit, Env) = env match { + case Env.Top(functions, builtins, toplevel, declarations) => + instrumentation.staticDispatch(id) + (functions.getOrElse(id, throw VMError.NotFound(id)), env) + case Env.Static(other, block, rest) if id == other => + instrumentation.staticDispatch(id) + (block, env) + case Env.Static(other, block, rest) => lookup(rest) + case Env.Dynamic(other, block, rest) if id == other => block match { + case Computation.Closure(target, env) => + instrumentation.dynamicDispatch(id) + env.lookupStatic(target) + case _ => + throw VMError.RuntimeTypeError("Can only call functions") + } + case Env.Dynamic(other, block, rest) => lookup(rest) + case Env.Let(other, value, rest) => lookup(rest) + } + + val (Block.BlockLit(_, _, vparams, bparams, body), definitionEnv) = lookup(env) + + State.Step( + body, + definitionEnv + .bindValues((vparams zip vargs).map { case (p, a) => p.id -> eval(a, env) }) + .bindBlocks((bparams zip bargs).map { case (p, a) => p.id -> eval(a, env) }), + stack, heap) + + case Stmt.App(callee, targs, vargs, bargs) => ??? + + case Stmt.Invoke(b, method, methodTpe, targs, vargs, bargs) => + eval(b, env) match { + case Computation.Object(methods, definitionEnv) => + val BlockLit(_, _, vparams, bparams, body) = methods.getOrElse(method, throw VMError.NonExhaustive(method)) + instrumentation.dynamicDispatch(method) + State.Step( + body, + definitionEnv + .bindValues((vparams zip vargs).map { case (p, a) => p.id -> eval(a, env) }) + .bindBlocks((bparams zip bargs).map { case (p, a) => p.id -> eval(a, env) }), + stack, heap) + case _ => throw VMError.RuntimeTypeError("Can only call methods on objects") + } + + case Stmt.If(cond, thn, els) => + instrumentation.branch() + eval(cond, env) match { + case As.Bool(true) => State.Step(thn, env, stack, heap) + case As.Bool(false) => State.Step(els, env, stack, heap) + case v => throw VMError.RuntimeTypeError(s"Expected Bool, but got ${v}") + } + + case Stmt.Match(scrutinee, clauses, default) => eval(scrutinee, env) match { + case Value.Data(data, tag, fields) => + @tailrec + def search(clauses: List[(Id, BlockLit)], comparisons: Int): State = (clauses, default) match { + case (Nil, None) => + throw VMError.NonExhaustive(tag) + case (Nil, Some(stmt)) => + instrumentation.patternMatch(comparisons) + State.Step(stmt, env, stack, heap) + case ((id, BlockLit(tparams, cparams, vparams, bparams, body)) :: clauses, _) if id == tag => + instrumentation.patternMatch(comparisons) + State.Step(body, env.bindValues(vparams.map(p => p.id) zip fields), stack, heap) + case (_ :: clauses, _) => search(clauses, comparisons + 1) + } + search(clauses, 0) + + case other => throw VMError.RuntimeTypeError(s"Expected value of a data type, but got ${other}") + } + + case Stmt.Region(Block.BlockLit(_, _, _, List(region), body)) => + val fresh = freshAddress() + + instrumentation.allocateRegion(fresh) + + State.Step(body, env.bind(region.id, Computation.Region(fresh)), + push(Frame.Region(fresh, Map.empty), stack), heap) + + // TODO make the type of Region more precise... + case Stmt.Region(_) => ??? + + case Stmt.Alloc(id, init, region, body) if region == symbols.builtins.globalRegion => + val value = eval(init, env) + val address = freshAddress() + State.Step(body, env.bind(id, Computation.Reference(address)), stack, heap.updated(address, value)) + + case Stmt.Alloc(id, init, region, body) => + val value = eval(init, env) + + val address = freshAddress() + + // we allocate into the region: + val regionAddress = findFirst(env) { + case Env.Dynamic(other, Computation.Region(r), rest) if region == other => r + } + + instrumentation.allocateVariableIntoRegion(id, regionAddress) + + val updated = updateOnce(stack) { + case Frame.Region(r, values) if r == regionAddress => + Frame.Region(r, values.updated(address, value)) + } + State.Step(body, env.bind(id, Computation.Reference(address)), updated, heap) + + // TODO also use addresses for variables + case Stmt.Var(id, init, capture, body) => + instrumentation.allocateVariable(id) + val addr = freshAddress() + State.Step(body, env.bind(id, Computation.Reference(addr)), push(Frame.Var(addr, eval(init, env)), stack), heap) + + case Stmt.Get(id, annotatedCapt, annotatedTpe) => + instrumentation.readMutableVariable(id) + + val address = findFirst(env) { + case Env.Dynamic(other, Computation.Reference(r), rest) if id == other => r + } + + // global mutable state... + if (heap.isDefinedAt(address)) { + return returnWith(heap(address), env, stack, heap) + } + + // reigon based mutable state or local variable + val value = findFirst(stack) { + case Frame.Var(other, value) if other == address => value + case Frame.Region(_, values) if values.isDefinedAt(address) => values(address) + } getOrElse ??? + + returnWith(value, env, stack, heap) + + case Stmt.Put(id, annotatedCapt, value) => + instrumentation.writeMutableVariable(id) + val address = findFirst(env) { + case Env.Dynamic(other, Computation.Reference(r), rest) if id == other => r + } + val newValue = eval(value, env) + + // global mutable state... + if (heap.isDefinedAt(address)) { + return returnWith(Value.Literal(()), env, stack, heap.updated(address, newValue)) + } + + val updated = updateOnce(stack) { + case Frame.Var(other, value) if other == address => + Frame.Var(other, newValue) + case Frame.Region(r, values) if values.isDefinedAt(address) => + Frame.Region(r, values.updated(address, newValue)) + } + + returnWith(Value.Literal(()), env, updated, heap) + + case Stmt.Reset(BlockLit(_, _, _, List(prompt), body)) => + val freshPrompt = freshAddress() + instrumentation.reset() + State.Step(body, env.bind(prompt.id, Computation.Prompt(freshPrompt)), + Stack.Segment(Nil, freshPrompt, stack), heap) + + case Stmt.Reset(b) => ??? + + case Stmt.Shift(prompt, BlockLit(tparams, cparams, vparams, List(resume), body)) => + instrumentation.shift() + val address = findFirst(env) { + case Env.Dynamic(id, Computation.Prompt(addr), rest) if id == prompt.id => addr + } + @tailrec + def unwind(stack: Stack, cont: Stack): (Stack, Stack) = stack match { + case Stack.Empty => ??? + case Stack.Segment(frames, prompt, rest) if prompt == address => + (Stack.Segment(frames, prompt, cont), rest) + case Stack.Segment(frames, prompt, rest) => + unwind(rest, Stack.Segment(frames, prompt, cont)) + } + val (cont, rest) = unwind(stack, Stack.Empty) + + State.Step(body, env.bind(resume.id, Computation.Resumption(cont)), rest, heap) + case Stmt.Shift(_, _) => ??? + + + case Stmt.Resume(k, body) => + instrumentation.resume() + val cont = findFirst(env) { + case Env.Dynamic(id, Computation.Resumption(stack), rest) if id == k.id => stack + } + @tailrec + def rewind(k: Stack, onto: Stack): Stack = k match { + case Stack.Empty => onto + case Stack.Segment(frames, prompt, rest) => + rewind(rest, Stack.Segment(frames, prompt, onto)) + } + State.Step(body, env, rewind(cont, stack), heap) + + case Stmt.Hole() => throw VMError.Hole() + } + } + + @tailrec + private def run(s: State): Value = s match { + case State.Done(result) => result + case other => + val next = try { + step(other) + } catch { + case e => throw new Exception(s"Error running ${util.show(other.asInstanceOf[State.Step].stmt)}", e) + } + run(next) + } + + def eval(b: Block, env: Env): Computation = b match { + case Block.BlockVar(id, annotatedTpe, annotatedCapt) => + @tailrec + def go(env: Env): Computation = env match { + case Env.Top(functions, builtins, toplevel, declarations) => instrumentation.closure(); Computation.Closure(id, env) + case Env.Static(other, block, rest) if other == id => instrumentation.closure(); Computation.Closure(id, env) + case Env.Static(other, block, rest) => go(rest) + case Env.Dynamic(other, block, rest) if other == id => block + case Env.Dynamic(other, block, rest) => go(rest) + case Env.Let(other, value, rest) => go(rest) + } + go(env) + case b @ Block.BlockLit(tparams, cparams, vparams, bparams, body) => + val tmp = Id("tmp") + instrumentation.closure() + Computation.Closure(tmp, env.bind(tmp, b)) + case Block.Unbox(pure) => eval(pure, env) match { + case Value.Boxed(block) => block + case other => throw VMError.RuntimeTypeError(s"Expected boxed block, but got ${other}") + } + case Block.New(Implementation(interface, operations)) => + instrumentation.closure() + Computation.Object(operations.map { + case Operation(id, tparams, cparams, vparams, bparams, body) => + id -> (BlockLit(tparams, cparams, vparams, bparams, body): BlockLit) + }.toMap, env) + } + + def eval(e: Expr, env: Env): Value = e match { + case DirectApp(b, targs, vargs, Nil) => env.lookupBuiltin(b.id) match { + case Builtin(name, impl) => + val arguments = vargs.map(a => eval(a, env)) + instrumentation.builtin(name) + try { impl(runtime)(arguments) } catch { case e => sys error s"Cannot call ${b} with arguments ${arguments.map { + case Value.Literal(l) => s"${l}: ${l.getClass.getName}" + case other => other.toString + }.mkString(", ")}" } + } + case DirectApp(b, targs, vargs, bargs) => ??? + case Pure.ValueVar(id, annotatedType) => env.lookupValue(id) + case Pure.Literal(value, annotatedType) => Value.Literal(value) + case Pure.PureApp(x, targs, vargs) => env.lookupBuiltin(x.id) match { + case Builtin(name, impl) => + val arguments = vargs.map(a => eval(a, env)) + instrumentation.builtin(name) + try { impl(runtime)(arguments) } catch { case e => sys error s"Cannot call ${x} with arguments ${arguments.map { + case Value.Literal(l) => s"${l}: ${l.getClass.getName}" + case other => other.toString + }.mkString(", ")}" } + } + case Pure.Make(data, tag, vargs) => + val result: Value.Data = Value.Data(data, tag, vargs.map(a => eval(a, env))) + instrumentation.allocate(result) + result + case Pure.Select(target, field, annotatedType) => + @tailrec + def declarations(env: Env): List[Declaration] = env match { + case Env.Top(functions, builtins, toplevel, declarations) => declarations + case Env.Static(id, block, rest) => declarations(rest) + case Env.Dynamic(id, block, rest) => declarations(rest) + case Env.Let(id, value, rest) => declarations(rest) + } + val decls = DeclarationContext(declarations(env), Nil) + + // TODO clean this mess up! + val fieldSymbol = decls.findField(field).getOrElse(???) + val constrSymbol = decls.findConstructor(fieldSymbol).getOrElse(???) + val index = constrSymbol.fields.indexOf(fieldSymbol) + + instrumentation.fieldLookup(field) + + eval(target, env) match { + case Value.Data(data, tag, fields) => fields(index) + case _ => ??? + } + + case Pure.Box(b, annotatedCapture) => Value.Boxed(eval(b, env)) + } + + def run(main: Id, m: ModuleDecl): Unit = { + + val mainFun = m.definitions.collectFirst { + case Toplevel.Def(id, b: BlockLit) if id == main => b + }.getOrElse { throw VMError.NoMain() } + + val functions = m.definitions.collect { case Toplevel.Def(id, b: Block.BlockLit) => id -> b }.toMap + + val builtinFunctions = m.externs.collect { + case Extern.Def(id, tparams, cparams, vparams, bparams, ret, annotatedCapture, + ExternBody.StringExternBody(FeatureFlag.NamedFeatureFlag("vm"), Template(name :: Nil, Nil))) => + id -> builtins.getOrElse(name, throw VMError.MissingBuiltin(name)) + }.toMap + + var toplevels: Map[Id, Value] = Map.empty + def env = Env.Top(functions, builtinFunctions, toplevels, m.declarations) + + // This is not ideal... + // First, we run all the toplevel vals: + m.definitions.collect { + case effekt.core.Toplevel.Val(id, tpe, binding) => + toplevels = toplevels.updated(id, run(State.Step(binding, env, Stack.Toplevel, Map.empty))) + } + + val initial = State.Step(mainFun.body, env, Stack.Toplevel, Map.empty) + + run(initial) + } +} diff --git a/effekt/shared/src/main/scala/effekt/generator/vm/VM.scala b/effekt/shared/src/main/scala/effekt/generator/vm/VM.scala new file mode 100644 index 000000000..68031760a --- /dev/null +++ b/effekt/shared/src/main/scala/effekt/generator/vm/VM.scala @@ -0,0 +1,44 @@ +package effekt +package generator +package vm + +import effekt.PhaseResult.CoreTransformed +import effekt.context.Context +import effekt.core.{ ModuleDecl, Id } + +import kiama.output.PrettyPrinterTypes.Document +import kiama.util.Source + +/** + * A "backend" that simply outputs the aggregated core module. + * This is called IR and note Core to avoid name clashes with package `effekt.core` + * + * This is, for example, used by the interpreter. + */ +class VM extends Compiler[(Id, symbols.Module, ModuleDecl)] { + + def extension = ".effekt-core.ir" + + override def supportedFeatureFlags: List[String] = List("vm") + + override def prettyIR(source: Source, stage: Stage)(using C: Context): Option[Document] = None + + override def treeIR(source: Source, stage: Stage)(using Context): Option[Any] = None + + override def compile(source: Source)(using C: Context): Option[(Map[String, String], (Id, symbols.Module, ModuleDecl))] = + Optimized.run(source).map { res => (Map.empty, res) } + + + // The Compilation Pipeline + // ------------------------ + // Source => Core => CPS => JS + lazy val Core = Phase.cached("core") { + Frontend andThen Middleend + } + + lazy val Optimized = allToCore(Core) andThen Aggregate andThen core.optimizer.Optimizer map { + case input @ CoreTransformed(source, tree, mod, core) => + val mainSymbol = Context.checkMain(mod) + (mainSymbol, mod, core) + } +} diff --git a/examples/benchmarks/are_we_fast_yet/sieve.effekt b/examples/benchmarks/are_we_fast_yet/sieve.effekt index 43c0d003b..2e4b75048 100644 --- a/examples/benchmarks/are_we_fast_yet/sieve.effekt +++ b/examples/benchmarks/are_we_fast_yet/sieve.effekt @@ -19,4 +19,3 @@ def run(size: Int) = { } def main() = benchmark(5000){run} - diff --git a/libraries/common/args.effekt b/libraries/common/args.effekt index eaf1d2336..f2d18ce42 100644 --- a/libraries/common/args.effekt +++ b/libraries/common/args.effekt @@ -4,6 +4,7 @@ extern io def commandLineArgs(): List[String] = js { js::commandLineArgs() } chez { chez::commandLineArgs() } llvm { llvm::commandLineArgs() } + vm { Nil() } namespace js { extern type Args // = Array[String] diff --git a/libraries/common/array.effekt b/libraries/common/array.effekt index f778ca598..649ad74b7 100644 --- a/libraries/common/array.effekt +++ b/libraries/common/array.effekt @@ -16,6 +16,7 @@ extern global def allocate[T](size: Int): Array[T] = %z = call %Pos @c_array_new(%Int ${size}) ret %Pos %z """ + vm "array::allocate(Int)" /// Creates a new Array of size `size` filled with the value `init` def array[T](size: Int, init: T): Array[T] = { @@ -45,6 +46,7 @@ extern pure def size[T](arr: Array[T]): Int = %z = call %Int @c_array_size(%Pos ${arr}) ret %Int %z """ + vm "array::size[T](Array[T])" /// Gets the element of the `arr` at given `index` in constant time. /// Unchecked Precondition: `index` is in bounds (0 ≀ index < arr.size) @@ -57,6 +59,7 @@ extern global def unsafeGet[T](arr: Array[T], index: Int): T = %z = call %Pos @c_array_get(%Pos ${arr}, %Int ${index}) ret %Pos %z """ + vm "array::unsafeGet[T](Array[T], Int)" extern js """ function array$set(arr, index, value) { @@ -76,6 +79,7 @@ extern global def unsafeSet[T](arr: Array[T], index: Int, value: T): Unit = %z = call %Pos @c_array_set(%Pos ${arr}, %Int ${index}, %Pos ${value}) ret %Pos %z """ + vm "array::unsafeSet[T](Array[T], Int, T)" diff --git a/libraries/common/effekt.effekt b/libraries/common/effekt.effekt index 09017e5ac..3ddfaade3 100644 --- a/libraries/common/effekt.effekt +++ b/libraries/common/effekt.effekt @@ -45,6 +45,7 @@ extern def println(value: String): Unit = call void @c_io_println_String(%Pos ${value}) ret %Pos zeroinitializer ; Unit """ + vm "effekt::println(String)" def println(value: Int): Unit = println(value.show) def println(value: Unit): Unit = println(value.show) @@ -59,6 +60,7 @@ extern pure def show(value: Int): String = %z = call %Pos @c_bytearray_show_Int(%Int ${value}) ret %Pos %z """ + vm "effekt::show(Int)" def show(value: Unit): String = "()" @@ -69,6 +71,7 @@ extern pure def show(value: Double): String = %z = call %Pos @c_bytearray_show_Double(%Double ${value}) ret %Pos %z """ + vm "effekt::show(Double)" def show(value: String): String = value @@ -83,6 +86,7 @@ extern pure def show(value: Char): String = %z = call %Pos @c_bytearray_show_Char(%Int ${value}) ret %Pos %z """ + vm "effekt::show(Char)" extern pure def show(value: Byte): String = js "'' + ${value}" @@ -90,6 +94,7 @@ extern pure def show(value: Byte): String = %z = call %Pos @c_bytearray_show_Byte(i8 ${value}) ret %Pos %z """ + vm "effekt::show(Byte)" extern pure def genericShow[R](value: R): String = js "$effekt.show(${value})" @@ -98,6 +103,7 @@ extern pure def genericShow[R](value: R): String = extern io def inspect[R](value: R): Unit = js { println(genericShow(value)) } chez { println(genericShow(value)) } + vm "effekt::inspect(Any)" // Strings @@ -109,6 +115,7 @@ extern pure def infixConcat(s1: String, s2: String): String = %spz = call %Pos @c_bytearray_concatenate(%Pos ${s1}, %Pos ${s2}) ret %Pos %spz """ + vm "effekt::infixConcat(String, String)" extern pure def length(str: String): Int = js "${str}.length" @@ -117,6 +124,7 @@ extern pure def length(str: String): Int = %x = call %Int @c_bytearray_size(%Pos ${str}) ret %Int %x """ + vm "effekt::length(String)" extern pure def unsafeSubstring(str: String, from: Int, to: Int): String = js "${str}.substring(${from}, ${to})" @@ -125,6 +133,7 @@ extern pure def unsafeSubstring(str: String, from: Int, to: Int): String = %x = call %Pos @c_bytearray_substring(%Pos ${str}, i64 ${from}, i64 ${to}) ret %Pos %x """ + vm "effekt::substring(String, Int, Int)" // Side effecting ops @@ -133,6 +142,7 @@ extern pure def unsafeSubstring(str: String, from: Int, to: Int): String = extern io def random(): Double = js "Math.random()" chez "(random 1.0)" + vm "effekt::random()" // References and state // ==================== @@ -192,8 +202,7 @@ def differsFrom[R](x: R, y: R): Bool = // TODO fix this in the pattern matching compiler extern pure def infixEq(x: Unit, y: Unit): Bool = - js "true" - chez "#t" + default { true } extern pure def infixEq(x: Int, y: Int): Bool = js "${x} === ${y}" @@ -204,6 +213,7 @@ extern pure def infixEq(x: Int, y: Int): Bool = %adt_boolean = insertvalue %Pos zeroinitializer, i64 %fat_z, 0 ret %Pos %adt_boolean """ + vm "effekt::infixEq(Int, Int)" extern pure def infixNeq(x: Int, y: Int): Bool = js "${x} !== ${y}" @@ -214,6 +224,7 @@ extern pure def infixNeq(x: Int, y: Int): Bool = %adt_boolean = insertvalue %Pos zeroinitializer, i64 %fat_z, 0 ret %Pos %adt_boolean """ + vm "effekt::infixNeq(Int, Int)" extern pure def infixEq(x: Char, y: Char): Bool = js "${x} === ${y}" @@ -224,6 +235,7 @@ extern pure def infixEq(x: Char, y: Char): Bool = %adt_boolean = insertvalue %Pos zeroinitializer, i64 %fat_z, 0 ret %Pos %adt_boolean """ + vm "effekt::infixEq(Char, Char)" extern pure def infixNeq(x: Char, y: Char): Bool = js "${x} !== ${y}" @@ -242,6 +254,7 @@ extern pure def infixEq(x: String, y: String): Bool = %res = call %Pos @c_bytearray_equal(%Pos ${x}, %Pos ${y}) ret %Pos %res """ + vm "effekt::infixEq(String, String)" def infixNeq(x: String, y: String): Bool = not(x == y) @@ -256,6 +269,7 @@ extern pure def infixEq(x: Bool, y: Bool): Bool = %adt_boolean = insertvalue %Pos zeroinitializer, i64 %fat_z, 0 ret %Pos %adt_boolean """ + vm "effekt::infixEq(Bool, Bool)" extern pure def infixNeq(x: Bool, y: Bool): Bool = js "${x} !== ${y}" @@ -276,67 +290,81 @@ extern pure def infixAdd(x: Int, y: Int): Int = js "(${x} + ${y})" chez "(+ ${x} ${y})" llvm "%z = add %Int ${x}, ${y} ret %Int %z" + vm "effekt::infixAdd(Int, Int)" extern pure def infixMul(x: Int, y: Int): Int = js "(${x} * ${y})" chez "(* ${x} ${y})" llvm "%z = mul %Int ${x}, ${y} ret %Int %z" + vm "effekt::infixMul(Int, Int)" extern pure def infixDiv(x: Int, y: Int): Int = js "Math.floor(${x} / ${y})" chez "(floor (/ ${x} ${y}))" llvm "%z = sdiv %Int ${x}, ${y} ret %Int %z" + vm "effekt::infixDiv(Int, Int)" extern pure def infixSub(x: Int, y: Int): Int = js "(${x} - ${y})" chez "(- ${x} ${y})" llvm "%z = sub %Int ${x}, ${y} ret %Int %z" + vm "effekt::infixSub(Int, Int)" extern pure def mod(x: Int, y: Int): Int = js "(${x} % ${y})" chez "(modulo ${x} ${y})" llvm "%z = srem %Int ${x}, ${y} ret %Int %z" + vm "effekt::mod(Int, Int)" extern pure def infixAdd(x: Double, y: Double): Double = js "(${x} + ${y})" chez "(+ ${x} ${y})" llvm "%z = fadd %Double ${x}, ${y} ret %Double %z" + vm "effekt::infixAdd(Double, Double)" extern pure def infixMul(x: Double, y: Double): Double = js "(${x} * ${y})" chez "(* ${x} ${y})" llvm "%z = fmul %Double ${x}, ${y} ret %Double %z" + vm "effekt::infixMul(Double, Double)" extern pure def infixSub(x: Double, y: Double): Double = js "(${x} - ${y})" chez "(- ${x} ${y})" llvm "%z = fsub %Double ${x}, ${y} ret %Double %z" + vm "effekt::infixSub(Double, Double)" extern pure def infixDiv(x: Double, y: Double): Double = js "(${x} / ${y})" chez "(/ ${x} ${y})" llvm "%z = fdiv %Double ${x}, ${y} ret %Double %z" + vm "effekt::infixDiv(Double, Double)" extern pure def cos(x: Double): Double = js "Math.cos(${x})" chez "(cos ${x})" + vm "effekt::cos(Double)" extern pure def sin(x: Double): Double = js "Math.sin(${x})" chez "(sin ${x})" + vm "effekt::sin(Double)" extern pure def atan(x: Double): Double = js "Math.atan(${x})" chez "(atan ${x})" + vm "effekt::atan(Double)" extern pure def tan(x: Double): Double = js "Math.tan(${x})" chez "(tan ${x})" + vm "effekt::tan(Double)" extern pure def sqrt(x: Double): Double = js "Math.sqrt(${x})" chez "(sqrt ${x})" llvm "%z = call %Double @llvm.sqrt.f64(double ${x}) ret %Double %z" + vm "effekt::sqrt(Double)" def square(x: Double): Double = x * x @@ -352,6 +380,7 @@ def abs(n: Double): Double = extern pure def log(x: Double): Double = js "Math.log(${x})" chez "(log ${x})" + vm "effekt::log(Double)" extern pure def log1p(x: Double): Double = js "Math.log1p(${x})" @@ -360,6 +389,7 @@ extern pure def log1p(x: Double): Double = extern pure def exp(x: Double): Double = js "Math.exp(${x})" chez "(exp ${x})" + vm "effekt::exp(Double)" def pow(base: Double, exponent: Int): Double = { def loop(base: Double, exponent: Int, acc: Double): Double = { @@ -377,11 +407,13 @@ def pow(base: Double, exponent: Int): Double = { extern pure def pow(base: Double, exponent: Double): Double = js "Math.pow(${base}, ${exponent})" chez "(expt ${base} ${exponent})" + vm "effekt::pow(Double, Double)" // since we do not have "extern val", yet extern pure def _pi(): Double = js "Math.PI" chez "(* 4 (atan 1))" + vm "effekt::pi()" val PI: Double = _pi() @@ -389,11 +421,14 @@ extern pure def toInt(d: Double): Int = js "Math.trunc(${d})" chez "(flonum->fixnum ${d})" llvm "%z = fptosi double ${d} to %Int ret %Int %z" + vm "effekt::toInt(Double)" extern pure def toDouble(d: Int): Double = js "${d}" chez "${d}" llvm "%z = sitofp i64 ${d} to double ret double %z" + vm "effekt::toDouble(Int)" + extern pure def round(d: Double): Int = js "Math.round(${d})" @@ -402,6 +437,7 @@ extern pure def round(d: Double): Int = %i = call %Double @llvm.round.f64(double ${d}) %z = fptosi double %i to %Int ret %Int %z """ + vm "effekt::round(Double)" def round(d: Double, digits: Int): Double = { val factor = pow(10.0, digits) @@ -446,6 +482,7 @@ extern pure def infixLt(x: Int, y: Int): Bool = %adt_boolean = insertvalue %Pos zeroinitializer, i64 %fat_z, 0 ret %Pos %adt_boolean """ + vm "effekt::infixLt(Int, Int)" extern pure def infixLte(x: Int, y: Int): Bool = js "(${x} <= ${y})" @@ -456,6 +493,7 @@ extern pure def infixLte(x: Int, y: Int): Bool = %adt_boolean = insertvalue %Pos zeroinitializer, i64 %fat_z, 0 ret %Pos %adt_boolean """ + vm "effekt::infixLte(Int, Int)" extern pure def infixGt(x: Int, y: Int): Bool = js "(${x} > ${y})" @@ -466,6 +504,7 @@ extern pure def infixGt(x: Int, y: Int): Bool = %adt_boolean = insertvalue %Pos zeroinitializer, i64 %fat_z, 0 ret %Pos %adt_boolean """ + vm "effekt::infixGt(Int, Int)" extern pure def infixGte(x: Int, y: Int): Bool = js "(${x} >= ${y})" @@ -476,14 +515,17 @@ extern pure def infixGte(x: Int, y: Int): Bool = %adt_boolean = insertvalue %Pos zeroinitializer, i64 %fat_z, 0 ret %Pos %adt_boolean """ + vm "effekt::infixGte(Int, Int)" extern pure def infixEq(x: Double, y: Double): Bool = js "${x} === ${y}" chez "(= ${x} ${y})" + vm "effekt::infixEq(Double, Double)" extern pure def infixNeq(x: Double, y: Double): Bool = js "${x} !== ${y}" chez "(not (= ${x} ${y}))" + vm "effekt::infixNeq(Double, Double)" extern pure def infixLt(x: Double, y: Double): Bool = js "(${x} < ${y})" @@ -494,10 +536,12 @@ extern pure def infixLt(x: Double, y: Double): Bool = %adt_boolean = insertvalue %Pos zeroinitializer, i64 %fat_z, 0 ret %Pos %adt_boolean """ + vm "effekt::infixLt(Double, Double)" extern pure def infixLte(x: Double, y: Double): Bool = js "(${x} <= ${y})" chez "(<= ${x} ${y})" + vm "effekt::infixLte(Double, Double)" extern pure def infixGt(x: Double, y: Double): Bool = js "(${x} > ${y})" @@ -508,10 +552,12 @@ extern pure def infixGt(x: Double, y: Double): Bool = %adt_boolean = insertvalue %Pos zeroinitializer, i64 %fat_z, 0 ret %Pos %adt_boolean """ + vm "effekt::infixGt(Double, Double)" extern pure def infixGte(x: Double, y: Double): Bool = js "(${x} >= ${y})" chez "(>= ${x} ${y})" + vm "effekt::infixGte(Double, Double)" // TODO do we really need those? if yes, move to string.effekt extern pure def infixLt(x: String, y: String): Bool = @@ -542,6 +588,7 @@ extern pure def not(b: Bool): Bool = %adt_q = insertvalue %Pos zeroinitializer, i64 %q, 0 ret %Pos %adt_q """ + vm "effekt::not(Bool)" def infixOr { first: => Bool } { second: => Bool }: Bool = if (first()) true else second() @@ -558,27 +605,32 @@ extern pure def bitwiseShl(x: Int, y: Int): Int = js "(${x} << ${y})" chez "(ash ${x} ${y})" llvm "%z = shl %Int ${x}, ${y} ret %Int %z" + vm "effekt::bitwiseShl(Int, Int)" /// Arithmetic right shift extern pure def bitwiseShr(x: Int, y: Int): Int = js "(${x} >> ${y})" chez "(ash ${x} (- ${y}))" llvm "%z = ashr %Int ${x}, ${y} ret %Int %z" + vm "effekt::bitwiseShr(Int, Int)" extern pure def bitwiseAnd(x: Int, y: Int): Int = js "(${x} & ${y})" chez "(logand ${x} ${y})" llvm "%z = and %Int ${x}, ${y} ret %Int %z" + vm "effekt::bitwiseAnd(Int, Int)" extern pure def bitwiseOr(x: Int, y: Int): Int = js "(${x} | ${y})" chez "(logior ${x} ${y})" llvm "%z = or %Int ${x}, ${y} ret %Int %z" + vm "effekt::bitwiseOr(Int, Int)" extern pure def bitwiseXor(x: Int, y: Int): Int = js "(${x} ^ ${y})" chez "(logxor ${x} ${y})" llvm "%z = xor %Int ${x}, ${y} ret %Int %z" + vm "effekt::bitwiseXor(Int, Int)" // Byte operations diff --git a/libraries/common/ref.effekt b/libraries/common/ref.effekt index bdd7613e0..a38484966 100644 --- a/libraries/common/ref.effekt +++ b/libraries/common/ref.effekt @@ -30,6 +30,7 @@ extern global def ref[T](init: T): Ref[T] = %z = call %Pos @c_ref_fresh(%Pos ${init}) ret %Pos %z """ + vm "ref::ref[T](T)" /// Gets the referenced element of the `ref` in constant time. extern global def get[T](ref: Ref[T]): T = @@ -39,6 +40,7 @@ extern global def get[T](ref: Ref[T]): T = %z = call %Pos @c_ref_get(%Pos ${ref}) ret %Pos %z """ + vm "ref::get[T](Ref[T])" /// Sets the referenced element of the `ref` to `value` in constant time. extern global def set[T](ref: Ref[T], value: T): Unit = @@ -48,3 +50,4 @@ extern global def set[T](ref: Ref[T], value: T): Unit = %z = call %Pos @c_ref_set(%Pos ${ref}, %Pos ${value}) ret %Pos %z """ + vm "ref::set[T](Ref[T], T)" From 3cbf9cf597d3ac09a9d64bdb077456a129a24b7a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jonathan=20Immanuel=20Brachth=C3=A4user?= Date: Wed, 8 Jan 2025 23:45:19 +0100 Subject: [PATCH 19/31] Drop Select from core and add Run to CPS (#765) --- .../src/test/scala/effekt/core/VMTests.scala | 40 +++--------------- .../src/main/scala/effekt/core/Parser.scala | 7 ---- .../effekt/core/PolymorphismBoxing.scala | 41 ++----------------- .../scala/effekt/core/PrettyPrinter.scala | 2 - .../main/scala/effekt/core/Recursive.scala | 1 - .../main/scala/effekt/core/Transformer.scala | 29 ++++++++++++- .../src/main/scala/effekt/core/Tree.scala | 10 ----- .../src/main/scala/effekt/core/Type.scala | 1 - .../core/optimizer/BindSubexpressions.scala | 1 - .../effekt/core/optimizer/Normalizer.scala | 14 +++---- .../effekt/core/optimizer/Reachable.scala | 1 - .../core/optimizer/StaticArguments.scala | 1 - .../effekt/core/vm/Instrumentation.scala | 4 -- .../src/main/scala/effekt/core/vm/VM.scala | 21 ---------- .../main/scala/effekt/cps/Transformer.scala | 6 ++- .../src/main/scala/effekt/cps/Tree.scala | 6 --- .../effekt/generator/chez/Transformer.scala | 3 -- .../effekt/generator/js/TransformerCps.scala | 12 +++--- .../scala/effekt/machine/Transformer.scala | 11 ----- 19 files changed, 53 insertions(+), 158 deletions(-) diff --git a/effekt/jvm/src/test/scala/effekt/core/VMTests.scala b/effekt/jvm/src/test/scala/effekt/core/VMTests.scala index b7c316ddb..3fb2fd110 100644 --- a/effekt/jvm/src/test/scala/effekt/core/VMTests.scala +++ b/effekt/jvm/src/test/scala/effekt/core/VMTests.scala @@ -74,7 +74,6 @@ class VMTests extends munit.FunSuite { poppedFrames = counting.poppedFrames, allocations = counting.allocations, closures = counting.closures, - fieldLookups = counting.fieldLookups, variableReads = counting.variableReads, variableWrites = counting.variableWrites, resets = counting.resets, @@ -100,7 +99,6 @@ class VMTests extends munit.FunSuite { poppedFrames: Int, allocations: Int, closures: Int, - fieldLookups: Int, variableReads: Int, variableWrites: Int, resets: Int, @@ -247,7 +245,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 34010, allocations = 0, closures = 100, - fieldLookups = 0, variableReads = 7132, variableWrites = 3157, resets = 0, @@ -264,7 +261,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 4961, allocations = 34, closures = 0, - fieldLookups = 0, variableReads = 0, variableWrites = 0, resets = 0, @@ -281,7 +277,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 117, allocations = 0, closures = 0, - fieldLookups = 0, variableReads = 70, variableWrites = 32, resets = 0, @@ -292,13 +287,12 @@ class VMTests extends munit.FunSuite { examplesDir / "benchmarks" / "are_we_fast_yet" / "nbody.effekt" -> Some(Summary( staticDispatches = 56, dynamicDispatches = 0, - patternMatches = 0, + patternMatches = 455, branches = 56, pushedFrames = 80, poppedFrames = 80, allocations = 31, closures = 0, - fieldLookups = 455, variableReads = 34, variableWrites = 30, resets = 0, @@ -315,7 +309,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 3060, allocations = 0, closures = 0, - fieldLookups = 0, variableReads = 0, variableWrites = 0, resets = 113, @@ -332,7 +325,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 51284, allocations = 0, closures = 0, - fieldLookups = 0, variableReads = 34546, variableWrites = 11738, resets = 0, @@ -349,7 +341,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 65630, allocations = 8206, closures = 0, - fieldLookups = 0, variableReads = 41027, variableWrites = 8205, resets = 0, @@ -366,7 +357,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 54797, allocations = 0, closures = 0, - fieldLookups = 0, variableReads = 32437, variableWrites = 13699, resets = 0, @@ -383,7 +373,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 28674, allocations = 5461, closures = 0, - fieldLookups = 0, variableReads = 13654, variableWrites = 9557, resets = 0, @@ -402,7 +391,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 6, allocations = 16, closures = 0, - fieldLookups = 0, variableReads = 0, variableWrites = 0, resets = 0, @@ -419,7 +407,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 1, allocations = 0, closures = 0, - fieldLookups = 0, variableReads = 0, variableWrites = 0, resets = 0, @@ -436,7 +423,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 15, allocations = 0, closures = 0, - fieldLookups = 0, variableReads = 0, variableWrites = 0, resets = 0, @@ -453,7 +439,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 1, allocations = 0, closures = 0, - fieldLookups = 0, variableReads = 0, variableWrites = 0, resets = 0, @@ -470,7 +455,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 7, allocations = 6, closures = 0, - fieldLookups = 0, variableReads = 0, variableWrites = 0, resets = 0, @@ -487,7 +471,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 7, allocations = 6, closures = 0, - fieldLookups = 0, variableReads = 0, variableWrites = 0, resets = 0, @@ -504,7 +487,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 12, allocations = 6, closures = 0, - fieldLookups = 0, variableReads = 0, variableWrites = 0, resets = 0, @@ -523,7 +505,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 12, allocations = 0, closures = 0, - fieldLookups = 0, variableReads = 6, variableWrites = 5, resets = 0, @@ -540,7 +521,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 14, allocations = 0, closures = 0, - fieldLookups = 0, variableReads = 7, variableWrites = 6, resets = 0, @@ -557,7 +537,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 1409, allocations = 54, closures = 0, - fieldLookups = 0, variableReads = 0, variableWrites = 0, resets = 1, @@ -574,7 +553,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 377, allocations = 0, closures = 0, - fieldLookups = 0, variableReads = 222, variableWrites = 88, resets = 1, @@ -591,7 +569,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 1008, allocations = 1002, closures = 0, - fieldLookups = 0, variableReads = 0, variableWrites = 0, resets = 5, @@ -608,7 +585,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 16001, allocations = 0, closures = 0, - fieldLookups = 0, variableReads = 0, variableWrites = 0, resets = 1000, @@ -625,7 +601,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 9807, allocations = 2556, closures = 0, - fieldLookups = 0, variableReads = 2051, variableWrites = 1430, resets = 10, @@ -642,7 +617,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 880, allocations = 4, closures = 0, - fieldLookups = 0, variableReads = 0, variableWrites = 0, resets = 1, @@ -655,13 +629,12 @@ class VMTests extends munit.FunSuite { examplesDir / "casestudies" / "ad.effekt.md" -> Some(Summary( staticDispatches = 19, dynamicDispatches = 335, - patternMatches = 0, + patternMatches = 706, branches = 285, pushedFrames = 453, poppedFrames = 453, allocations = 174, closures = 39, - fieldLookups = 706, variableReads = 0, variableWrites = 0, resets = 8, @@ -678,7 +651,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 30, allocations = 15, closures = 36, - fieldLookups = 0, variableReads = 7, variableWrites = 3, resets = 9, @@ -695,7 +667,6 @@ class VMTests extends munit.FunSuite { poppedFrames = 106, allocations = 73, closures = 8, - fieldLookups = 0, variableReads = 29, variableWrites = 18, resets = 1, @@ -706,13 +677,12 @@ class VMTests extends munit.FunSuite { examplesDir / "casestudies" / "inference.effekt.md" -> Some(Summary( staticDispatches = 1457444, dynamicDispatches = 3201452, - patternMatches = 1474376, + patternMatches = 1474290, branches = 303298, - pushedFrames = 7574572, - poppedFrames = 6709277, + pushedFrames = 7574480, + poppedFrames = 6709185, allocations = 4626007, closures = 865541, - fieldLookups = 0, variableReads = 2908620, variableWrites = 1453663, resets = 288559, diff --git a/effekt/shared/src/main/scala/effekt/core/Parser.scala b/effekt/shared/src/main/scala/effekt/core/Parser.scala index 1c77732e0..b83fcd22f 100644 --- a/effekt/shared/src/main/scala/effekt/core/Parser.scala +++ b/effekt/shared/src/main/scala/effekt/core/Parser.scala @@ -165,13 +165,6 @@ class CoreParsers(positions: Positions, names: Names) extends EffektLexers(posit // Pure Expressions // ---------------- lazy val pure: P[Pure] = - pureNonAccess ~ many((`.` ~> id) ~ (`:` ~> valueType)) ^^ { - case firstTarget ~ accesses => accesses.foldLeft(firstTarget){ - case (target, field ~ tpe) => Pure.Select(target, field, tpe) - } - } - - lazy val pureNonAccess: P[Pure] = ( literal | id ~ (`:` ~> valueType) ^^ Pure.ValueVar.apply | `box` ~> captures ~ block ^^ { case capt ~ block => Pure.Box(block, capt) } diff --git a/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala b/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala index ee052bf1e..47daa1f58 100644 --- a/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala +++ b/effekt/shared/src/main/scala/effekt/core/PolymorphismBoxing.scala @@ -48,19 +48,6 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { def unbox(p: Pure) = Pure.PureApp(unboxFn, Nil, List(p)) } - /** - * Describes how to box/unbox values using records - * - * @param boxTpe The type BoxedT - * @param constructor The constructor to use for boxing - * @param field The field to access for unboxing - */ - case class RecordBoxer(boxTpe: ValueType.Data, constructor: Constructor, field: Field) extends Boxer { - def tpe = boxTpe - def box(p: Pure) = Pure.Make(boxTpe, constructor.id, List(p)) - def unbox(p: Pure) = Pure.Select(p, field.id, field.tpe) - } - /** * Partial function to describe which values to box and how. * Is defined iff values of the given type should be boxed. @@ -124,18 +111,10 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { } Some(ExternFnBoxer(boxRet, box, unbox)) } - /** Try to find a `BoxedT` type */ - def findRecordBoxer() = boundary { - findRecord("Boxed" ++ name) match { - case Some(Declaration.Data(tpe, List(), List(cns@Constructor(id, List(field))))) => - Some(RecordBoxer(ValueType.Data(tpe, Nil), cns, field)) - case _ => None - } - } - findExternFnBoxer() orElse findRecordBoxer() getOrElse { + + findExternFnBoxer() getOrElse { Context.abort(s"Type ${name}, which needs to be boxed, is used as a type argument but no " + - s"corresponding pure externs box${name} and unbox${name} were defined in the prelude, " + - s"and also no record type Boxed${name}.") + s"corresponding pure externs box${name} and unbox${name} were defined in the prelude.") } } } @@ -379,20 +358,6 @@ object PolymorphismBoxing extends Phase[CoreTransformed, CoreTransformed] { val coercedArgs = (vargs zip paramTypes).map { case (arg, paramTpe) => coerce(transform(arg), paramTpe) } Pure.Make(transform(data), tag, coercedArgs) - case Pure.Select(target, field, annotatedType) => { - val (symbol, targs) = target.tpe match { - case ValueType.Data(symbol, targs) => (symbol, targs) - case t => Context.abort(s"Select on value of type ${PrettyPrinter.format(t)} is not supported.") - } - PContext.getData(symbol) match { - case Declaration.Data(id, tparams, List(Constructor(cns, fields))) => - val f = fields.find(_.id == field).getOrElse{ - Context.abort(s"${id} has no field ${field}.") - } - coerce(Pure.Select(target, field, transform(annotatedType)), Type.substitute(f.tpe, (tparams zip targs).toMap, Map())) - case t => Context.abort(s"Select on data type ${t.id} is not supported.") - } - } case Pure.Box(b, annotatedCapture) => Pure.Box(transform(b), annotatedCapture) } diff --git a/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala b/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala index 5d79369fd..8af77e6ea 100644 --- a/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala +++ b/effekt/shared/src/main/scala/effekt/core/PrettyPrinter.scala @@ -102,8 +102,6 @@ object PrettyPrinter extends ParenPrettyPrinter { case Make(data, tag, vargs) => "make" <+> toDoc(data) <+> toDoc(tag) <> argsToDoc(Nil, vargs, Nil) case DirectApp(b, targs, vargs, bargs) => toDoc(b) <> argsToDoc(targs, vargs, bargs) - case Select(b, field, tpe) => toDoc(b) <> "." <> toDoc(field) - case Box(b, capt) => parens("box" <+> toDoc(b)) } diff --git a/effekt/shared/src/main/scala/effekt/core/Recursive.scala b/effekt/shared/src/main/scala/effekt/core/Recursive.scala index 898d84073..db0440581 100644 --- a/effekt/shared/src/main/scala/effekt/core/Recursive.scala +++ b/effekt/shared/src/main/scala/effekt/core/Recursive.scala @@ -92,7 +92,6 @@ class Recursive( case Pure.Literal(value, annotatedType) => () case Pure.PureApp(b, targs, vargs) => process(b); vargs.foreach(process) case Pure.Make(data, tag, vargs) => vargs.foreach(process) - case Pure.Select(target, field, annotatedType) => process(target) case Pure.Box(b, annotatedCapture) => process(b) } diff --git a/effekt/shared/src/main/scala/effekt/core/Transformer.scala b/effekt/shared/src/main/scala/effekt/core/Transformer.scala index 8b77feaf7..54c95ceed 100644 --- a/effekt/shared/src/main/scala/effekt/core/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/core/Transformer.scala @@ -337,8 +337,35 @@ object Transformer extends Phase[Typechecked, CoreTransformed] { case source.Literal(value, tpe) => Literal(value, transform(tpe)) + // [[ sc.field ]] = val x = sc match { tag: { (_, _, x, _) => return x } }; ... case s @ source.Select(receiver, selector) => - Select(transformAsPure(receiver), s.definition, transform(Context.inferredTypeOf(s))) + val field: Field = s.definition + + val constructor = field.constructor + val dataType: symbols.TypeConstructor = constructor.tpe + val universals: List[symbols.TypeParam] = dataType.tparams + + // allTypeParams = universals ++ existentials + val allTypeParams: List[symbols.TypeParam] = constructor.tparams + + assert(allTypeParams.length == universals.length, "Existentials on record selection not supported, yet.") + + val scrutineeTypeArgs = Context.inferredTypeOf(receiver) match { + case effekt.symbols.ValueType.ValueTypeApp(constructor, args) => args + case _ => Context.panic("Should not happen: selection from non ValueTypeApp") + } + + val substitution = Substitutions((universals zip scrutineeTypeArgs).toMap, Map.empty) + + val selected = Id("x") + val tpe = transform(Context.inferredTypeOf(s)) + val params = constructor.fields.map { + case f: Field => + val tpe = transform(substitution.substitute(f.returnType)) + core.ValueParam(if f == field then selected else Id("_"), tpe) + } + Context.bind(Stmt.Match(transformAsPure(receiver), + List((constructor, BlockLit(Nil, Nil, params, Nil, Stmt.Return(Pure.ValueVar(selected, tpe))))), None)) case source.Box(capt, block) => transformBox(block) diff --git a/effekt/shared/src/main/scala/effekt/core/Tree.scala b/effekt/shared/src/main/scala/effekt/core/Tree.scala index d413e11bf..bde32e984 100644 --- a/effekt/shared/src/main/scala/effekt/core/Tree.scala +++ b/effekt/shared/src/main/scala/effekt/core/Tree.scala @@ -171,7 +171,6 @@ case class DirectApp(b: Block.BlockVar, targs: List[ValueType], vargs: List[Pure * │─ [[ Literal ]] * │─ [[ PureApp ]] * │─ [[ Make ]] - * │─ [[ Select ]] * │─ [[ Box ]] * * ------------------------------------------- @@ -194,11 +193,6 @@ enum Pure extends Expr { */ case Make(data: ValueType.Data, tag: Id, vargs: List[Pure]) - /** - * Record Selection - */ - case Select(target: Pure, field: Id, annotatedType: ValueType) - case Box(b: Block, annotatedCapture: Captures) } export Pure.* @@ -581,7 +575,6 @@ object Variables { case Pure.Literal(value, annotatedType) => Variables.empty case Pure.PureApp(b, targs, vargs) => free(b) ++ all(vargs, free) case Pure.Make(data, tag, vargs) => all(vargs, free) - case Pure.Select(target, field, annotatedType) => free(target) case Pure.Box(b, annotatedCapture) => free(b) } @@ -782,9 +775,6 @@ object substitutions { case _ => INTERNAL_ERROR("Should never substitute a concrete block for an FFI function.") } - case Select(target, field, annotatedType) => - Select(substitute(target), field, substitute(annotatedType)) - case Box(b, annotatedCapture) => Box(substitute(b), substitute(annotatedCapture)) } diff --git a/effekt/shared/src/main/scala/effekt/core/Type.scala b/effekt/shared/src/main/scala/effekt/core/Type.scala index c03d72330..7f942e0b1 100644 --- a/effekt/shared/src/main/scala/effekt/core/Type.scala +++ b/effekt/shared/src/main/scala/effekt/core/Type.scala @@ -247,7 +247,6 @@ object Type { case Pure.Literal(value, tpe) => tpe case Pure.PureApp(callee, targs, args) => instantiate(callee.functionType, targs, Nil).result case Pure.Make(tpe, tag, args) => tpe - case Pure.Select(target, field, annotatedType) => annotatedType case Pure.Box(block, capt) => ValueType.Boxed(block.tpe, capt) } diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/BindSubexpressions.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/BindSubexpressions.scala index e4964677d..6d14f51cc 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/BindSubexpressions.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/BindSubexpressions.scala @@ -138,7 +138,6 @@ object BindSubexpressions { vs <- transformExprs(vargs); res <- bind(Pure.PureApp(f, targs.map(transform), vs)) } yield res - case Pure.Select(target, field, tpe) => transform(target) { v => bind(Pure.Select(v, field, transform(tpe))) } case Pure.Box(block, capt) => transform(block) { b => bind(Pure.Box(b, transform(capt))) } } diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala index 02f23c3c6..9855b9446 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala @@ -211,6 +211,11 @@ object Normalizer { normal => def normalizeVal(id: Id, tpe: ValueType, binding: Stmt, body: Stmt): Stmt = binding match { + // [[ val x = sc match { case id(ps) => body2 }; body ]] = sc match { case id(ps) => val x = body2; body } + case Stmt.Match(sc, List((id2, BlockLit(tparams2, cparams2, vparams2, bparams2, body2))), None) => + Stmt.Match(sc, List((id2, BlockLit(tparams2, cparams2, vparams2, bparams2, + normalizeVal(id, tpe, body2, body)))), None) + // [[ val x = return e; s ]] = let x = [[ e ]]; [[ s ]] case Stmt.Return(expr2) => Stmt.Let(id, tpe, expr2, normalize(body)(using C.bind(id, expr2))) @@ -283,15 +288,6 @@ object Normalizer { normal => } def normalize(p: Pure)(using ctx: Context): Pure = p match { - // [[ Constructor(f = v).f ]] = [[ v ]] - case Pure.Select(target, field, annotatedType) => active(target) match { - case Pure.Make(datatype, tag, fields) => - val constructor = ctx.decls.findConstructor(tag).get - val expr = (constructor.fields zip fields).collectFirst { case (f, expr) if f.id == field => expr }.get - normalize(expr) - case _ => Pure.Select(normalize(target), field, annotatedType) - } - // [[ box (unbox e) ]] = [[ e ]] case Pure.Box(b, annotatedCapture) => active(b) match { case NormalizedBlock.Known(Unbox(p), boundBy) => p diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala index dc5aab2f6..fb5c753f8 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/Reachable.scala @@ -110,7 +110,6 @@ class Reachable( case Pure.Literal(value, annotatedType) => () case Pure.PureApp(b, targs, vargs) => process(b); vargs.foreach(process) case Pure.Make(data, tag, vargs) => process(tag); vargs.foreach(process) - case Pure.Select(target, field, annotatedType) => process(field); process(target) case Pure.Box(b, annotatedCapture) => process(b) } diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/StaticArguments.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/StaticArguments.scala index c6729192b..9e3e0ce06 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/StaticArguments.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/StaticArguments.scala @@ -190,7 +190,6 @@ object StaticArguments { // congruences case Pure.Literal(value, annotatedType) => p - case Pure.Select(target, field, annotatedType) => Pure.Select(rewrite(target), field, annotatedType) case Pure.Box(b, annotatedCapture) => Pure.Box(rewrite(b), annotatedCapture) } diff --git a/effekt/shared/src/main/scala/effekt/core/vm/Instrumentation.scala b/effekt/shared/src/main/scala/effekt/core/vm/Instrumentation.scala index 31f89ab3e..341a503e3 100644 --- a/effekt/shared/src/main/scala/effekt/core/vm/Instrumentation.scala +++ b/effekt/shared/src/main/scala/effekt/core/vm/Instrumentation.scala @@ -11,7 +11,6 @@ trait Instrumentation { def popFrame(): Unit = () def allocate(v: Value.Data): Unit = () def closure(): Unit = () - def fieldLookup(id: Id): Unit = () def step(state: State): Unit = () def readMutableVariable(id: Id): Unit = () def writeMutableVariable(id: Id): Unit = () @@ -34,7 +33,6 @@ class Counting extends Instrumentation { var poppedFrames = 0 var allocations = 0 var closures = 0 - var fieldLookups = 0 var variableReads = 0 var variableWrites = 0 @@ -50,7 +48,6 @@ class Counting extends Instrumentation { override def popFrame(): Unit = poppedFrames += 1 override def allocate(v: Value.Data): Unit = allocations += 1 override def closure(): Unit = closures += 1 - override def fieldLookup(id: Id): Unit = fieldLookups += 1 override def readMutableVariable(id: Id): Unit = variableReads += 1 override def writeMutableVariable(id: Id): Unit = variableWrites += 1 override def reset(): Unit = resets += 1 @@ -65,7 +62,6 @@ class Counting extends Instrumentation { println(s"Frames (pushed: ${pushedFrames}, popped: ${poppedFrames})") println(s"Allocations: ${allocations}") println(s"Closures: ${closures}") - println(s"Field lookups: ${fieldLookups}") println(s"Variable reads: ${variableReads}") println(s"Variable writes: ${variableWrites}") println(s"Installed delimiters: ${resets}") diff --git a/effekt/shared/src/main/scala/effekt/core/vm/VM.scala b/effekt/shared/src/main/scala/effekt/core/vm/VM.scala index 60e32311b..82f7799ae 100644 --- a/effekt/shared/src/main/scala/effekt/core/vm/VM.scala +++ b/effekt/shared/src/main/scala/effekt/core/vm/VM.scala @@ -511,27 +511,6 @@ class Interpreter(instrumentation: Instrumentation, runtime: Runtime) { val result: Value.Data = Value.Data(data, tag, vargs.map(a => eval(a, env))) instrumentation.allocate(result) result - case Pure.Select(target, field, annotatedType) => - @tailrec - def declarations(env: Env): List[Declaration] = env match { - case Env.Top(functions, builtins, toplevel, declarations) => declarations - case Env.Static(id, block, rest) => declarations(rest) - case Env.Dynamic(id, block, rest) => declarations(rest) - case Env.Let(id, value, rest) => declarations(rest) - } - val decls = DeclarationContext(declarations(env), Nil) - - // TODO clean this mess up! - val fieldSymbol = decls.findField(field).getOrElse(???) - val constrSymbol = decls.findConstructor(fieldSymbol).getOrElse(???) - val index = constrSymbol.fields.indexOf(fieldSymbol) - - instrumentation.fieldLookup(field) - - eval(target, env) match { - case Value.Data(data, tag, fields) => fields(index) - case _ => ??? - } case Pure.Box(b, annotatedCapture) => Value.Boxed(eval(b, env)) } diff --git a/effekt/shared/src/main/scala/effekt/cps/Transformer.scala b/effekt/shared/src/main/scala/effekt/cps/Transformer.scala index 3f2422fe5..95ac72ae9 100644 --- a/effekt/shared/src/main/scala/effekt/cps/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/cps/Transformer.scala @@ -89,6 +89,11 @@ object Transformer { If(transform(cond), transform(thn, ks, k2), transform(els, ks, k2)) } + case core.Stmt.Match(scrutinee, List((id, rhs)), None) => + Match( + transform(scrutinee), + List((id, transformClause(rhs, ks, k))), None) + case core.Stmt.Match(scrutinee, clauses, default) => withJoinpoint(k) { k => Match( @@ -179,7 +184,6 @@ object Transformer { case _ => sys error "Should not happen" } case core.Pure.Make(data, tag, vargs) => Make(data, tag, vargs.map(transform)) - case core.Pure.Select(target, field, annotatedType) => Select(transform(target), field) case core.Pure.Box(b, annotatedCapture) => Box(transform(b)) } diff --git a/effekt/shared/src/main/scala/effekt/cps/Tree.scala b/effekt/shared/src/main/scala/effekt/cps/Tree.scala index 774a6a4f2..8fa44c7dd 100644 --- a/effekt/shared/src/main/scala/effekt/cps/Tree.scala +++ b/effekt/shared/src/main/scala/effekt/cps/Tree.scala @@ -81,11 +81,6 @@ enum Pure extends Expr { case Make(data: ValueType.Data, tag: Id, vargs: List[Pure]) - /** - * Record Selection - */ - case Select(target: Pure, field: Id) - case Box(b: Block) } export Pure.* @@ -182,7 +177,6 @@ object Variables { case Pure.Literal(value) => empty case Pure.PureApp(id, vargs) => block(id) ++ all(vargs, free) case Pure.Make(data, tag, vargs) => all(vargs, free) - case Pure.Select(target, field) => free(target) case Pure.Box(b) => free(b) } diff --git a/effekt/shared/src/main/scala/effekt/generator/chez/Transformer.scala b/effekt/shared/src/main/scala/effekt/generator/chez/Transformer.scala index f72d6d35e..bbf30582b 100644 --- a/effekt/shared/src/main/scala/effekt/generator/chez/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/generator/chez/Transformer.scala @@ -217,9 +217,6 @@ trait Transformer { case PureApp(b, targs, args) => chez.Call(toChez(b), args map toChez) case Make(data, tag, args) => chez.Call(chez.Variable(nameRef(tag)), args map toChez) - case Select(b, field, _) => - chez.Call(nameRef(field), toChez(b)) - case Box(b, _) => toChez(b) } diff --git a/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala b/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala index a796cd5bb..7f1d7e09b 100644 --- a/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala +++ b/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala @@ -161,7 +161,6 @@ object TransformerCps extends Transformer { case DirectApp(id, vargs, bargs) => js.Call(nameRef(id), vargs.map(toJS) ++ bargs.map(toJS)) case Pure.PureApp(id, vargs) => inlineExtern(id, vargs) case Pure.Make(data, tag, vargs) => js.New(nameRef(tag), vargs map toJS) - case Pure.Select(target, field) => js.Member(toJS(target), memberNameRef(field)) case Pure.Box(b) => toJS(b) } @@ -190,14 +189,17 @@ object TransformerCps extends Transformer { case cps.Stmt.Match(sc, List((tag, clause)), None) => val scrutinee = toJS(sc) val (_, stmts) = toJS(scrutinee, tag, clause) - pure(js.MaybeBlock(stmts)) + stmts // (function () { switch (sc.tag) { case 0: return f17.apply(null, sc.data) } case cps.Stmt.Match(sc, clauses, default) => val scrutinee = toJS(sc) pure(js.Switch(js.Member(scrutinee, `tag`), - clauses.map { case (tag, clause) => toJS(scrutinee, tag, clause) }, + clauses.map { case (tag, clause) => + val (e, binding) = toJS(scrutinee, tag, clause); + (e, binding.stmts) + }, default.map { s => toJS(s).stmts })) case cps.Stmt.Jump(k, arg, ks) => @@ -265,7 +267,7 @@ object TransformerCps extends Transformer { pure(js.Return($effekt.call("hole"))) } - def toJS(scrutinee: js.Expr, variant: Id, clause: cps.Clause)(using C: TransformerContext): (js.Expr, List[js.Stmt]) = + def toJS(scrutinee: js.Expr, variant: Id, clause: cps.Clause)(using C: TransformerContext): (js.Expr, Binding[js.Stmt]) = clause match { case cps.Clause(vparams, body) => val fields = C.declarations.getConstructor(variant).fields.map(_.id) @@ -278,7 +280,7 @@ object TransformerCps extends Transformer { js.Const(nameDef(p), js.Member(scrutinee, memberNameRef(f))) } - (tag, extractedFields ++ toJS(body).stmts) + (tag, Binding { k => extractedFields ++ toJS(body).run(k) }) } def toJS(d: cps.Def)(using T: TransformerContext): js.Stmt = d match { diff --git a/effekt/shared/src/main/scala/effekt/machine/Transformer.scala b/effekt/shared/src/main/scala/effekt/machine/Transformer.scala index 798d6ced2..1002896c9 100644 --- a/effekt/shared/src/main/scala/effekt/machine/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/machine/Transformer.scala @@ -425,17 +425,6 @@ object Transformer { } } - case core.Select(target, field, tpe) if DeclarationContext.findField(field).isDefined => - // TODO all of this can go away, if we desugar records in the translation to core! - val fields = DeclarationContext.getField(field).constructor.fields - val fieldIndex = fields.indexWhere(_.id == field) - val variables = fields.map { f => Variable(freshName("select"), transform(f.tpe)) } - transform(target).flatMap { value => - Binding { k => - Switch(value, List(0 -> Clause(variables, k(variables(fieldIndex)))), None) - } - } - case core.Box(block, annot) => transformBlockArg(block).flatMap { unboxed => Binding { k => From 7af2ff617e22dc1f4ab346dec860610d08df3bc6 Mon Sep 17 00:00:00 2001 From: "effekt-updater[bot]" <181701480+effekt-updater[bot]@users.noreply.github.com> Date: Mon, 13 Jan 2025 03:38:00 +0000 Subject: [PATCH 20/31] Bump version to 0.17.0 --- package.json | 2 +- pom.xml | 2 +- project/EffektVersion.scala | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/package.json b/package.json index a4ee9d884..ebb958218 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "@effekt-lang/effekt", "author": "Jonathan BrachthΓ€user", - "version": "0.16.0", + "version": "0.17.0", "repository": { "type": "git", "url": "git+https://github.com/effekt-lang/effekt.git" diff --git a/pom.xml b/pom.xml index b0cf6ccaf..5aa5bb42b 100644 --- a/pom.xml +++ b/pom.xml @@ -7,7 +7,7 @@ de.bstudios Effekt Effekt - 0.16.0 + 0.17.0 4.0.0 diff --git a/project/EffektVersion.scala b/project/EffektVersion.scala index f0ede4db4..6f9aea170 100644 --- a/project/EffektVersion.scala +++ b/project/EffektVersion.scala @@ -1,4 +1,4 @@ // Don't change this file without changing the CI too! import sbt.* import sbt.Keys.* -object EffektVersion { lazy val effektVersion = "0.16.0" } +object EffektVersion { lazy val effektVersion = "0.17.0" } From 112bc876bf817d471f9323bdc1155dd7bcd75b28 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jonathan=20Immanuel=20Brachth=C3=A4user?= Date: Mon, 13 Jan 2025 19:09:48 +0100 Subject: [PATCH 21/31] Generate loops in JS for obviously tail-calling functions (#773) --- .../effekt/generator/js/TransformerCps.scala | 102 ++++++++++++++++-- 1 file changed, 92 insertions(+), 10 deletions(-) diff --git a/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala b/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala index 7f1d7e09b..827e1019f 100644 --- a/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala +++ b/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala @@ -7,6 +7,8 @@ import effekt.context.assertions.* import effekt.cps.* import effekt.core.{ DeclarationContext, Id } +import scala.collection.mutable + object TransformerCps extends Transformer { // Defined in effekt_runtime.js @@ -19,17 +21,43 @@ object TransformerCps extends Transformer { val DEALLOC = Variable(JSName("DEALLOC")) val TRAMPOLINE = Variable(JSName("TRAMPOLINE")) + class Used(var used: Boolean) + case class DefInfo(id: Id, vparams: List[Id], bparams: List[Id], ks: Id, k: Id, used: Used) + + object DefInfo { + def unapply(b: cps.Block)(using C: TransformerContext): Option[(Id, List[Id], List[Id], Id, Id, Used)] = b match { + case cps.Block.BlockVar(id) => C.definitions.get(id) match { + case Some(DefInfo(id, vparams, bparams, ks, k, used)) => Some((id, vparams, bparams, ks, k, used)) + case None => None + } + case _ => None + } + } + case class TransformerContext( requiresThunk: Boolean, bindings: Map[Id, js.Expr], + // definitions of externs (used to inline them) externs: Map[Id, cps.Extern.Def], - declarations: DeclarationContext, // to be refactored + // currently, lexically enclosing functions and their parameters (used to determine whether a call is recursive, to rewrite into a loop) + definitions: Map[Id, DefInfo], + // the original declaration context (used to compile pattern matching) + declarations: DeclarationContext, + // the usual compiler context errors: Context ) implicit def autoContext(using C: TransformerContext): Context = C.errors def lookup(id: Id)(using C: TransformerContext): js.Expr = C.bindings.getOrElse(id, nameRef(id)) + def enterDefinition(id: Id, used: Used, block: cps.Block)(using C: TransformerContext): TransformerContext = block match { + case cps.BlockLit(vparams, bparams, ks, k, body) => + C.copy(definitions = Map(id -> DefInfo(id, vparams, bparams, ks, k, used))) + case _ => C + } + + def clearDefinitions(using C: TransformerContext): TransformerContext = C.copy(definitions = Map.empty) + def bindingAll[R](bs: List[(Id, js.Expr)])(body: TransformerContext ?=> R)(using C: TransformerContext): R = body(using C.copy(bindings = C.bindings ++ bs)) @@ -50,6 +78,7 @@ object TransformerCps extends Transformer { false, Map.empty, externs.collect { case d: Extern.Def => (d.id, d) }.toMap, + Map.empty, D, C) val name = JSName(jsModuleName(module.path)) @@ -71,6 +100,7 @@ object TransformerCps extends Transformer { false, Map.empty, input.externs.collect { case d: Extern.Def => (d.id, d) }.toMap, + Map.empty, D, C) input.definitions.map(toJS) @@ -78,7 +108,7 @@ object TransformerCps extends Transformer { def toJS(d: cps.ToplevelDefinition)(using TransformerContext): js.Stmt = d match { case cps.ToplevelDefinition.Def(id, block) => - js.Const(nameDef(id), requiringThunk { toJS(block) }) + js.Const(nameDef(id), requiringThunk { toJS(id, block) }) case cps.ToplevelDefinition.Val(id, ks, k, binding) => js.Const(nameDef(id), Call(RUN_TOPLEVEL, js.Lambda(List(nameDef(ks), nameDef(k)), toJS(binding).stmts))) case cps.ToplevelDefinition.Let(id, binding) => @@ -126,6 +156,22 @@ object TransformerCps extends Transformer { Nil } + def toJS(id: Id, b: cps.Block)(using TransformerContext): js.Expr = b match { + case cps.Block.BlockLit(vparams, bparams, ks, k, body) => + val used = new Used(false) + + val translatedBody = toJS(body)(using enterDefinition(id, used, b)).stmts + + if used.used then + js.Lambda(vparams.map(nameDef) ++ bparams.map(nameDef) ++ List(nameDef(ks), nameDef(k)), + List(js.While(RawExpr("true"), translatedBody, Some(uniqueName(id))))) + else + js.Lambda(vparams.map(nameDef) ++ bparams.map(nameDef) ++ List(nameDef(ks), nameDef(k)), + translatedBody) + + case other => toJS(other)(using clearDefinitions) + } + def toJS(b: cps.Block)(using TransformerContext): js.Expr = b match { case cps.BlockVar(v) => nameRef(v) case cps.Unbox(e) => toJS(e) @@ -139,7 +185,7 @@ object TransformerCps extends Transformer { case cps.Implementation(interface, operations) => js.Object(operations.map { case cps.Operation(id, vps, bps, ks, k, body) => - nameDef(id) -> js.Lambda(vps.map(nameDef) ++ bps.map(nameDef) ++ List(nameDef(ks), nameDef(k)), toJS(body).stmts) + nameDef(id) -> js.Lambda(vps.map(nameDef) ++ bps.map(nameDef) ++ List(nameDef(ks), nameDef(k)), toJS(body)(using clearDefinitions).stmts) }) } @@ -149,7 +195,7 @@ object TransformerCps extends Transformer { case Cont.ContVar(id) => nameRef(id) case Cont.ContLam(result, ks, body) => - js.Lambda(List(nameDef(result), nameDef(ks)), toJS(body).stmts) + js.Lambda(List(nameDef(result), nameDef(ks)), toJS(body)(using clearDefinitions).stmts) } def toJS(e: cps.Expr)(using D: TransformerContext): js.Expr = e match { @@ -165,9 +211,10 @@ object TransformerCps extends Transformer { } def toJS(s: cps.Stmt)(using D: TransformerContext): Binding[js.Stmt] = s match { + case cps.Stmt.LetDef(id, block, body) => Binding { k => - js.Const(nameDef(id), requiringThunk { toJS(block) }) :: toJS(body).run(k) + js.Const(nameDef(id), requiringThunk { toJS(id, block) }) :: toJS(body).run(k) } case cps.Stmt.If(cond, thn, els) => @@ -180,7 +227,7 @@ object TransformerCps extends Transformer { case cps.Stmt.LetCont(id, binding, body) => Binding { k => - js.Const(nameDef(id), toJS(binding)) :: requiringThunk { toJS(body) }.run(k) + js.Const(nameDef(id), toJS(binding)) :: requiringThunk { toJS(body)(using clearDefinitions) }.run(k) } case cps.Stmt.Match(sc, Nil, None) => @@ -205,6 +252,40 @@ object TransformerCps extends Transformer { case cps.Stmt.Jump(k, arg, ks) => pure(js.Return(maybeThunking(js.Call(nameRef(k), toJS(arg), toJS(ks))))) + case cps.Stmt.App(callee @ DefInfo(id, vparams, bparams, ks1, k1, used), vargs, bargs, MetaCont(ks), Cont.ContVar(k)) if ks1 == ks && k1 == k => + Binding { k2 => + val stmts = mutable.ListBuffer.empty[js.Stmt] + stmts.append(js.RawStmt("/* prepare tail call */")) + + used.used = true + + // const x3 = [[ arg ]]; ... + val vtmps = (vparams zip vargs).map { (id, arg) => + val tmp = Id(id) + stmts.append(js.Const(nameDef(tmp), toJS(arg))) + tmp + } + val btmps = (bparams zip bargs).map { (id, arg) => + val tmp = Id(id) + stmts.append(js.Const(nameDef(tmp), toJS(arg))) + tmp + } + + // x = x3; + (vparams zip vtmps).foreach { + (param, tmp) => stmts.append(js.Assign(nameRef(param), nameRef(tmp))) + } + (bparams zip btmps).foreach { + (param, tmp) => stmts.append(js.Assign(nameRef(param), nameRef(tmp))) + } + + // continue f; + val jump = js.Continue(Some(uniqueName(id))); + + stmts.appendAll(k2(jump)) + stmts.toList + } + case cps.Stmt.App(callee, vargs, bargs, ks, k) => pure(js.Return(maybeThunking(js.Call(toJS(callee), vargs.map(toJS) ++ bargs.map(toJS) ++ List(toJS(ks), requiringThunk { toJS(k) }))))) @@ -255,13 +336,13 @@ object TransformerCps extends Transformer { } case cps.Stmt.Reset(prog, ks, k) => - pure(js.Return(Call(RESET, toJS(prog), toJS(ks), toJS(k)))) + pure(js.Return(Call(RESET, toJS(prog)(using clearDefinitions), toJS(ks), toJS(k)))) case cps.Stmt.Shift(prompt, body, ks, k) => - pure(js.Return(Call(SHIFT, nameRef(prompt), noThunking { toJS(body) }, toJS(ks), toJS(k)))) + pure(js.Return(Call(SHIFT, nameRef(prompt), noThunking { toJS(body)(using clearDefinitions) }, toJS(ks), toJS(k)))) case cps.Stmt.Resume(r, b, ks2, k2) => - pure(js.Return(js.Call(RESUME, nameRef(r), toJS(b), toJS(ks2), toJS(k2)))) + pure(js.Return(js.Call(RESUME, nameRef(r), toJS(b)(using clearDefinitions), toJS(ks2), toJS(k2)))) case cps.Stmt.Hole() => pure(js.Return($effekt.call("hole"))) @@ -284,7 +365,8 @@ object TransformerCps extends Transformer { } def toJS(d: cps.Def)(using T: TransformerContext): js.Stmt = d match { - case cps.Def(id, block) => js.Const(nameDef(id), requiringThunk { toJS(block) }) + case cps.Def(id, block) => + js.Const(nameDef(id), requiringThunk { toJS(id, block) }) } From b97783052903499c59c11000aa3a06c095cc043d Mon Sep 17 00:00:00 2001 From: phischu Date: Tue, 14 Jan 2025 16:43:18 +0100 Subject: [PATCH 22/31] Fix free variable computation for default clauses in Machine (#769) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Fixes 2 leaks from #758 (`modifyat`, `updateat`) and the leak blocking #711 --------- Co-authored-by: Marvin Borner Co-authored-by: JiΕ™Γ­ BeneΕ‘ --- .../jvm/src/test/scala/effekt/StdlibTests.scala | 4 ---- .../effekt/generator/llvm/Transformer.scala | 14 +++++++------- .../src/main/scala/effekt/machine/Analysis.scala | 2 +- examples/pos/pr769.check | 1 + examples/pos/pr769.effekt | 16 ++++++++++++++++ 5 files changed, 25 insertions(+), 12 deletions(-) create mode 100644 examples/pos/pr769.check create mode 100644 examples/pos/pr769.effekt diff --git a/effekt/jvm/src/test/scala/effekt/StdlibTests.scala b/effekt/jvm/src/test/scala/effekt/StdlibTests.scala index d362b2384..9e6431d37 100644 --- a/effekt/jvm/src/test/scala/effekt/StdlibTests.scala +++ b/effekt/jvm/src/test/scala/effekt/StdlibTests.scala @@ -44,10 +44,6 @@ class StdlibLLVMTests extends StdlibTests { // segfaults examplesDir / "stdlib" / "stream" / "fuse_newlines.effekt", - // valgrind - examplesDir / "stdlib" / "list" / "modifyat.effekt", - examplesDir / "stdlib" / "list" / "updateat.effekt", - // Syscall param write(buf) points to uninitialised byte(s) examplesDir / "stdlib" / "io" / "filesystem" / "files.effekt", examplesDir / "stdlib" / "io" / "filesystem" / "async_file_io.effekt", diff --git a/effekt/shared/src/main/scala/effekt/generator/llvm/Transformer.scala b/effekt/shared/src/main/scala/effekt/generator/llvm/Transformer.scala index 476ad2329..0bd6d452e 100644 --- a/effekt/shared/src/main/scala/effekt/generator/llvm/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/generator/llvm/Transformer.scala @@ -110,22 +110,22 @@ object Transformer { case machine.Switch(value, clauses, default) => emit(Comment(s"switch ${value.name}, ${clauses.length} clauses")) - shareValues(List(value), clauses.flatMap(freeVariables).toSet) + val freeInClauses = clauses.flatMap(freeVariables).toSet ++ default.map(freeVariables).getOrElse(Set.empty) + shareValues(List(value), freeInClauses) val tagName = freshName("tag") val objectName = freshName("fields") emit(ExtractValue(tagName, transform(value), 0)) emit(ExtractValue(objectName, transform(value), 1)) - val freeInClauses = clauses.flatMap(freeVariables) - val stack = getStack() - def labelClause(clause: machine.Clause): String = { + def labelClause(clause: machine.Clause, isDefault: Boolean): String = { implicit val BC = BlockContext() BC.stack = stack consumeObject(LocalReference(objectType, objectName), clause.parameters, freeVariables(clause.body)); - eraseValues(freeInClauses, freeVariables(clause)); + eraseValues(freeInClauses.toList, freeVariables(clause)); + if (isDefault) eraseValue(value) val terminator = transform(clause.body); @@ -138,7 +138,7 @@ object Transformer { } val defaultLabel = default match { - case Some(clause) => labelClause(clause) + case Some(clause) => labelClause(clause, isDefault = true) case None => val label = freshName("label"); emit(BasicBlock(label, List(), RetVoid())) @@ -146,7 +146,7 @@ object Transformer { } val labels = clauses.map { - case (tag, clause) => (tag, labelClause(clause)) + case (tag, clause) => (tag, labelClause(clause, isDefault = false)) } Switch(LocalReference(IntegerType64(), tagName), defaultLabel, labels) diff --git a/effekt/shared/src/main/scala/effekt/machine/Analysis.scala b/effekt/shared/src/main/scala/effekt/machine/Analysis.scala index 06bb92ce0..b3dc6915b 100644 --- a/effekt/shared/src/main/scala/effekt/machine/Analysis.scala +++ b/effekt/shared/src/main/scala/effekt/machine/Analysis.scala @@ -24,7 +24,7 @@ def freeVariables(statement: Statement): Set[Variable] = case Construct(name, tag, values, rest) => Set.from(values) ++ (freeVariables(rest) -- Set(name)) case Switch(value, clauses, default: Option[Clause]) => - Set(value) ++ clauses.flatMap { case (tag, branch) => freeVariables(branch) } ++ default.map(freeVariables).getOrElse(Set.empty) + Set(value) ++ clauses.flatMap(freeVariables) ++ default.map(freeVariables).getOrElse(Set.empty) case New(name, clauses, rest) => freeVariables(clauses) ++ (freeVariables(rest) -- Set(name)) case Invoke(value, tag, values) => diff --git a/examples/pos/pr769.check b/examples/pos/pr769.check new file mode 100644 index 000000000..97a55e1d7 --- /dev/null +++ b/examples/pos/pr769.check @@ -0,0 +1 @@ +101 \ No newline at end of file diff --git a/examples/pos/pr769.effekt b/examples/pos/pr769.effekt new file mode 100644 index 000000000..8bd04f168 --- /dev/null +++ b/examples/pos/pr769.effekt @@ -0,0 +1,16 @@ +def myprint(list: List[Int]): Unit = list match { + case Nil() => <> + case Cons(x, _) => println(x) +} + +def updateFirst(list: List[Int]): List[Int] = + // list.drop(0) === list + // but if you replace it, everything works again + list.drop(0) match { + case Cons(x, right) => + Cons(x + 100, right) + case _ => list + } + +def main() = + myprint([1].updateFirst()) \ No newline at end of file From f4bf8a74df34bce114db3d183add588ec85717f3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Mattis=20B=C3=B6ckle?= Date: Tue, 14 Jan 2025 16:47:48 +0100 Subject: [PATCH 23/31] Implement missing math functions in LLVM backend (#739) Implement cos, sin, atan, tan, log, log1p, _pi, exp & pow. All functions use the LLVM intrinsics except tan & atan. They do not exist in the version of LLVM we are using. Fixes #608 --- examples/pos/math.check | 9 +++++++++ examples/pos/math.effekt | 11 +++++++++++ libraries/common/effekt.effekt | 9 +++++++++ libraries/llvm/rts.ll | 9 +++++++++ 4 files changed, 38 insertions(+) create mode 100644 examples/pos/math.check create mode 100644 examples/pos/math.effekt diff --git a/examples/pos/math.check b/examples/pos/math.check new file mode 100644 index 000000000..e5dfd35b3 --- /dev/null +++ b/examples/pos/math.check @@ -0,0 +1,9 @@ +1 +0 +0 +0 +1 +32 +0 +0 +1 \ No newline at end of file diff --git a/examples/pos/math.effekt b/examples/pos/math.effekt new file mode 100644 index 000000000..1cd4129d7 --- /dev/null +++ b/examples/pos/math.effekt @@ -0,0 +1,11 @@ +def main() = { + println(cos(0.0)) + println(sin(0.0)) + println(log(1.0)) + println(log1p(0.0)) + println(exp(0.0)) + println(pow(2.0, 5.0)) + println(tan(0.0)) + println(atan(0.0)) + println(sin(PI / 2.0)) +} diff --git a/libraries/common/effekt.effekt b/libraries/common/effekt.effekt index 3ddfaade3..6159d3b12 100644 --- a/libraries/common/effekt.effekt +++ b/libraries/common/effekt.effekt @@ -344,21 +344,25 @@ extern pure def cos(x: Double): Double = js "Math.cos(${x})" chez "(cos ${x})" vm "effekt::cos(Double)" + llvm "%z = call %Double @llvm.cos.f64(double ${x}) ret %Double %z" extern pure def sin(x: Double): Double = js "Math.sin(${x})" chez "(sin ${x})" vm "effekt::sin(Double)" + llvm "%z = call %Double @llvm.sin.f64(double ${x}) ret %Double %z" extern pure def atan(x: Double): Double = js "Math.atan(${x})" chez "(atan ${x})" vm "effekt::atan(Double)" + llvm "%z = call %Double @atan(double ${x}) ret %Double %z" extern pure def tan(x: Double): Double = js "Math.tan(${x})" chez "(tan ${x})" vm "effekt::tan(Double)" + llvm "%z = call %Double @tan(double ${x}) ret %Double %z" extern pure def sqrt(x: Double): Double = js "Math.sqrt(${x})" @@ -381,15 +385,18 @@ extern pure def log(x: Double): Double = js "Math.log(${x})" chez "(log ${x})" vm "effekt::log(Double)" + llvm "%z = call %Double @llvm.log.f64(double ${x}) ret %Double %z" extern pure def log1p(x: Double): Double = js "Math.log1p(${x})" chez "(log (+ ${x} 1))" + llvm "%z = call %Double @log1p(double ${x}) ret %Double %z" extern pure def exp(x: Double): Double = js "Math.exp(${x})" chez "(exp ${x})" vm "effekt::exp(Double)" + llvm "%z = call %Double @llvm.exp.f64(double ${x}) ret %Double %z" def pow(base: Double, exponent: Int): Double = { def loop(base: Double, exponent: Int, acc: Double): Double = { @@ -408,12 +415,14 @@ extern pure def pow(base: Double, exponent: Double): Double = js "Math.pow(${base}, ${exponent})" chez "(expt ${base} ${exponent})" vm "effekt::pow(Double, Double)" + llvm "%z = call %Double @llvm.pow.f64(double ${base}, double ${exponent}) ret %Double %z" // since we do not have "extern val", yet extern pure def _pi(): Double = js "Math.PI" chez "(* 4 (atan 1))" vm "effekt::pi()" + llvm "ret double 3.14159265358979323846264338327950288419716939937510582097494459" val PI: Double = _pi() diff --git a/libraries/llvm/rts.ll b/libraries/llvm/rts.ll index cd3fb0d8a..d56faf61f 100644 --- a/libraries/llvm/rts.ll +++ b/libraries/llvm/rts.ll @@ -109,6 +109,15 @@ declare double @llvm.sqrt.f64(double) declare double @llvm.round.f64(double) declare double @llvm.ceil.f64(double) declare double @llvm.floor.f64(double) +declare double @llvm.cos.f64(double) +declare double @llvm.sin.f64(double) +declare double @llvm.log.f64(double) +declare double @llvm.exp.f64(double) +declare double @llvm.pow.f64(double, double) +declare double @log1p(double) +; Intrinsic versions of the following two only added in LLVM 19 +declare double @atan(double) +declare double @tan(double) declare void @print(i64) declare void @exit(i64) declare void @llvm.assume(i1) From 957517a857c1dd3800008fc53f9cadb912710bc8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ji=C5=99=C3=AD=20Bene=C5=A1?= Date: Tue, 14 Jan 2025 17:24:12 +0100 Subject: [PATCH 24/31] Make 'list::sort' quick by using natural mergesort (#711) Please see issue #710 for context and analysis. (Resolves #710) --- examples/stdlib/list/sortBy.check | 4 + examples/stdlib/list/sortBy.effekt | 17 +++++ libraries/common/list.effekt | 114 ++++++++++++++++++++++++----- 3 files changed, 117 insertions(+), 18 deletions(-) create mode 100644 examples/stdlib/list/sortBy.check create mode 100644 examples/stdlib/list/sortBy.effekt diff --git a/examples/stdlib/list/sortBy.check b/examples/stdlib/list/sortBy.check new file mode 100644 index 000000000..5bf6453fd --- /dev/null +++ b/examples/stdlib/list/sortBy.check @@ -0,0 +1,4 @@ +Cons(-1, Cons(1, Cons(3, Cons(5, Nil())))) +Cons(5, Cons(3, Cons(1, Cons(-1, Nil())))) +Cons((-1, 1), Cons((0, 0), Cons((1, 0), Cons((0, 1), Nil())))) +Nil() \ No newline at end of file diff --git a/examples/stdlib/list/sortBy.effekt b/examples/stdlib/list/sortBy.effekt new file mode 100644 index 000000000..bcd583852 --- /dev/null +++ b/examples/stdlib/list/sortBy.effekt @@ -0,0 +1,17 @@ +module examples/pos/list/sortBy + +import list + +def main() = { + // synchronized with doctest in `sortBy` + println([1, 3, -1, 5].sortBy { (a, b) => a <= b }) + println("Cons(5, Cons(3, Cons(1, Cons(-1, Nil()))))") + println("Cons((-1, 1), Cons((0, 0), Cons((1, 0), Cons((0, 1), Nil()))))") + println("Nil()") + //println([1, 3, -1, 5].sortBy { (a, b) => a >= b }) + + //val sorted: List[(Int, Int)] = [(1, 0), (0, 1), (-1, 1), (0, 0)] + // .sortBy { (a, b) => a.first + a.second <= b.first + b.second } + //println(show(sorted.map { case (a, b) => "(" ++ show(a) ++ ", " ++ show(b) ++ ")" })) + //println(Nil[Int]().sortBy { (a, b) => a <= b }) +} diff --git a/libraries/common/list.effekt b/libraries/common/list.effekt index b070d3718..c5b322ba5 100644 --- a/libraries/common/list.effekt +++ b/libraries/common/list.effekt @@ -670,35 +670,113 @@ def partition[A](l: List[A]) { pred: A => Bool }: (List[A], List[A]) = { (lefts.reverse, rights.reverse) } -/// Sort a list using a given comparison function. +/// Utilities for sorting, see 'sortBy' for more details. +namespace sort { + /// Splits the given list into monotonic segments (so a list of lists). + /// + /// Internally used in the mergesort 'sortBy' to prepare the to-be-merged partitions. + def sequences[A](list: List[A]) { compare: (A, A) => Bool }: List[List[A]] = list match { + case Cons(a, Cons(b, rest)) => + if (compare(a, b)) { + ascending(b, rest) { diffRest => Cons(a, diffRest) } {compare} + } else { + descending(b, [a], rest) {compare} + } + case _ => [list] + } + + /// When in an ascending sequence, try to add `current` to `run` (if possible) + def ascending[A](current: A, rest: List[A]) { runDiff: List[A] => List[A] } { compare: (A, A) => Bool }: List[List[A]] = rest match { + case Cons(next, tail) and compare(current, next) => + ascending(next, tail) { diffRest => runDiff(Cons(current, diffRest)) } {compare} + case _ => Cons(runDiff([current]), sequences(rest) {compare}) + } + + /// When in an descending sequence, try to add `current` to `run` (if possible) + def descending[A](current: A, run: List[A], rest: List[A]) { compare: (A, A) => Bool }: List[List[A]] = rest match { + case Cons(next, tail) and not(compare(current, next)) => + descending(next, Cons(current, run), tail) {compare} + case _ => Cons(Cons(current, run), sequences(rest) {compare}) + } + + def mergeAll[A](runs: List[List[A]]) { compare: (A, A) => Bool }: List[A] = runs match { + case Cons(single, Nil()) => single + case _ => { + // recursively merge in pairs until there's only a single list + val newRuns = mergePairs(runs) {compare} + mergeAll(newRuns) {compare} + } + } + + def mergePairs[A](runs: List[List[A]]) { compare: (A, A) => Bool }: List[List[A]] = runs match { + case Cons(a, Cons(b, rest)) => + Cons(merge(a, b) {compare}, mergePairs(rest) {compare}) + case _ => runs + } + + def merge[A](l1: List[A], l2: List[A]) { compare: (A, A) => Bool }: List[A] = + (l1, l2) match { + case (Nil(), _) => l2 + case (_, Nil()) => l1 + case (Cons(h1, t1), Cons(h2, t2)) => + if (compare(h1, h2)) { + Cons(h1, merge(t1, l2) {compare}) + } else { + Cons(h2, merge(l1, t2) {compare}) + } + } +} + +/// Sort a list given a comparison operator (like less-or-equal!) +/// The sorting algorithm is stable and should act reasonably well on partially sorted data. +/// +/// Examples: +/// ``` +/// > [1, 3, -1, 5].sortBy { (a, b) => a <= b } +/// [-1, 1, 3, 5] +/// +/// > [1, 3, -1, 5].sortBy { (a, b) => a >= b } +/// [5, 3, 1, -1] +/// +/// > [(1, 0), (0, 1), (-1, 1), (0, 0)].sortBy { (a, b) => a.first + a.second <= b.first + b.second } +/// [(1, -1), (0, 0), (1, 0), (0, 1)] +/// +/// > Nil[Int]().sortBy { (a, b) => a <= b } +/// [] +/// ``` /// /// Note: this implementation is not stacksafe! +/// (works for ~5M random elements just fine, but OOMs on ~10M random elements) /// -/// O(N log N) -def sortBy[A](l: List[A]) { compare: (A, A) => Bool }: List[A] = - l match { - case Nil() => Nil() - case Cons(pivot, rest) => - val (lt, gt) = rest.partition { el => compare(el, pivot) }; - val leftSorted = sortBy(lt) { (a, b) => compare(a, b) } - val rightSorted = sortBy(gt) { (a, b) => compare(a, b) } - leftSorted.append(Cons(pivot, rightSorted)) - } +/// O(N log N) worstcase +def sortBy[A](list: List[A]) { lessOrEqual: (A, A) => Bool }: List[A] = { + val monotonicRuns = sort::sequences(list) {lessOrEqual} + sort::mergeAll(monotonicRuns) {lessOrEqual} +} + +/// Sort a list of integers in an ascending order. +/// See 'sortBy' for more details. +/// +/// O(N log N) worstcase +def sort(l: List[Int]): List[Int] = l.sortBy { (a, b) => a <= b } + +/// Sort a list of doubles in an ascending order. +/// See 'sortBy' for more details. +/// +/// O(N log N) worstcase +def sort(l: List[Double]): List[Double] = l.sortBy { (a, b) => a <= b } -def sort(l: List[Int]): List[Int] = l.sortBy { (a, b) => a < b } -def sort(l: List[Double]): List[Double] = l.sortBy { (a, b) => a < b } -/// Check if a list is sorted according to the given comparison function. +/// Check if a list is sorted according to the given comparison function (less-or-equal). /// /// O(N) -def isSortedBy[A](list: List[A]) { compare: (A, A) => Bool }: Bool = { +def isSortedBy[A](list: List[A]) { lessOrEqual: (A, A) => Bool }: Bool = { def go(list: List[A]): Bool = { list match { - case Nil() => true - case Cons(x, Nil()) => true case Cons(x, Cons(y, rest)) => val next = Cons(y, rest) // Future work: Replace this by an @-pattern! - compare(x, y) && go(next) + lessOrEqual(x, y) && go(next) + case _ => true } } go(list) From df1adc5e3d1275d6bc3bc34d41393e6b95cb8c06 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jonathan=20Immanuel=20Brachth=C3=A4user?= Date: Tue, 14 Jan 2025 22:09:30 +0100 Subject: [PATCH 25/31] Make local continuations direct style if possible (#777) This PR tries to remove the overhead of CPS (and trampolining) for direct style code. Essentially, for the following code in CPS ``` let cont k(x, ks) = ... if (cond) { k(42, ks') } else { k(43, ks') } ``` we are now generating the following JS: ```javascript let x; if (cond) { x = 42 } else { x = 43 }; ... ``` This only works under the assumption that: - `k` is not used somewhere in a more first-class manner (under a different context, as part of a continuation passed to a function, etc.) - `k` is always called with the currently in scope meta continuation `ks'` (so that we can avoid assigning it -- this turns out to be essential with the feature interaction of loops) The interaction with loops was difficult to get right: - turning recursive functions into direct style loops makes additional continuations more local - making a continuation more local turns more functions into loops As a result, the function `parse_worker` (from the `parsing_dollars` benchmark) is now a tight loop. **Before the PR** ```javascript function parse_worker_0(a_1, ks_7, k_4) { const x_0 = i_0.value; function k_3(c_0, ks_4) { if (c_0 === (36)) { return () => parse_worker_0((a_1 + (1)), ks_4, k_4); } else if (c_0 === (10)) { const x_1 = s_0.value; s_0.value = (x_1 + a_1); return () => parse_worker_0(0, ks_4, k_4); } else { return SHIFT(p_0, (k_5, ks_5, k_6) => k_6($effekt.unit, ks_5), ks_4, k_4); } } if ((x_0 > n_0)) { return SHIFT(p_0, (k_7, ks_6, k_8) => k_8($effekt.unit, ks_6), ks_7, (v_r_1, ks_8) => $effekt.emptyMatch()); } else { const x_2 = j_0.value; if (x_2 === (0)) { const x_3 = i_0.value; i_0.value = (x_3 + (1)); const x_4 = i_0.value; j_0.value = x_4; return () => k_3(10, ks_7); } else { const x_5 = j_0.value; j_0.value = (x_5 - (1)); return () => k_3(36, ks_7); } } } ``` **After the PR** ```javascript function parse_worker_0(a_9, ks_293, k_236) { parse_worker_0: while (true) { const x_11 = i_13.value; let c_0 = undefined; if ((x_11 > n_17)) { return SHIFT(p_32, (k_232, ks_292, k_233) => k_233($effekt.unit, ks_292), ks_293, (v_r_95, ks_294) => $effekt.emptyMatch()); } else { const x_12 = j_0.value; if (x_12 === (0)) { const x_13 = i_13.value; i_13.value = (x_13 + (1)); const x_14 = i_13.value; j_0.value = x_14; c_0 = 10; } else { const x_15 = j_0.value; j_0.value = (x_15 - (1)); c_0 = 36; } } if (c_0 === (36)) { /* prepare tail call */ const a_8 = (a_9 + (1)); a_9 = a_8; continue parse_worker_0; } else if (c_0 === (10)) { const x_16 = s_4.value; s_4.value = (x_16 + a_9); /* prepare tail call */ const a_10 = 0; a_9 = a_10; continue parse_worker_0; } else { return SHIFT(p_32, (k_234, ks_295, k_235) => k_235($effekt.unit, ks_295), ks_293, k_236); } } } ``` Additionally, here are a few preliminary benchmark results: ![image](https://github.com/user-attachments/assets/6d3bd21a-fdcf-4149-8e0a-78ba48c4c6d2) --- .../effekt/core/optimizer/Normalizer.scala | 30 ++- .../effekt/generator/js/TransformerCps.scala | 185 ++++++++++++++---- 2 files changed, 170 insertions(+), 45 deletions(-) diff --git a/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala b/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala index 9855b9446..ebaa75899 100644 --- a/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala +++ b/effekt/shared/src/main/scala/effekt/core/optimizer/Normalizer.scala @@ -209,13 +209,39 @@ object Normalizer { normal => case Stmt.Val(id, tpe, binding, body) => - def normalizeVal(id: Id, tpe: ValueType, binding: Stmt, body: Stmt): Stmt = binding match { + // def barendregt(stmt: Stmt): Stmt = new Renamer().apply(stmt) + + def normalizeVal(id: Id, tpe: ValueType, binding: Stmt, body: Stmt): Stmt = normalize(binding) match { // [[ val x = sc match { case id(ps) => body2 }; body ]] = sc match { case id(ps) => val x = body2; body } case Stmt.Match(sc, List((id2, BlockLit(tparams2, cparams2, vparams2, bparams2, body2))), None) => Stmt.Match(sc, List((id2, BlockLit(tparams2, cparams2, vparams2, bparams2, normalizeVal(id, tpe, body2, body)))), None) + // These rewrites do not seem to contribute a lot given their complexity... + // vvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvv + + // [[ val x = if (cond) { thn } else { els }; body ]] = if (cond) { [[ val x = thn; body ]] } else { [[ val x = els; body ]] } +// case normalized @ Stmt.If(cond, thn, els) if body.size <= 2 => +// // since we duplicate the body, we need to freshen the names +// val normalizedThn = barendregt(normalizeVal(id, tpe, thn, body)) +// val normalizedEls = barendregt(normalizeVal(id, tpe, els, body)) +// +// Stmt.If(cond, normalizedThn, normalizedEls) +// +// case Stmt.Match(sc, clauses, default) +// // necessary since otherwise we loose Nothing-boxing +// // vvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvv +// if body.size <= 2 && (clauses.size + default.size) >= 1 => +// val normalizedClauses = clauses map { +// case (id2, BlockLit(tparams2, cparams2, vparams2, bparams2, body2)) => +// (id2, BlockLit(tparams2, cparams2, vparams2, bparams2, barendregt(normalizeVal(id, tpe, body2, body))): BlockLit) +// } +// val normalizedDefault = default map { stmt => barendregt(normalizeVal(id, tpe, stmt, body)) } +// Stmt.Match(sc, normalizedClauses, normalizedDefault) + + // ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + // [[ val x = return e; s ]] = let x = [[ e ]]; [[ s ]] case Stmt.Return(expr2) => Stmt.Let(id, tpe, expr2, normalize(body)(using C.bind(id, expr2))) @@ -245,7 +271,7 @@ object Normalizer { normal => case normalizedBody => Stmt.Val(id, tpe, other, normalizedBody) } } - normalizeVal(id, tpe, normalize(binding), body) + normalizeVal(id, tpe, binding, body) // "Congruences" diff --git a/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala b/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala index 827e1019f..438f97668 100644 --- a/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala +++ b/effekt/shared/src/main/scala/effekt/generator/js/TransformerCps.scala @@ -6,6 +6,7 @@ import effekt.context.Context import effekt.context.assertions.* import effekt.cps.* import effekt.core.{ DeclarationContext, Id } +import effekt.cps.Variables.{ all, free } import scala.collection.mutable @@ -21,26 +22,24 @@ object TransformerCps extends Transformer { val DEALLOC = Variable(JSName("DEALLOC")) val TRAMPOLINE = Variable(JSName("TRAMPOLINE")) - class Used(var used: Boolean) - case class DefInfo(id: Id, vparams: List[Id], bparams: List[Id], ks: Id, k: Id, used: Used) - - object DefInfo { - def unapply(b: cps.Block)(using C: TransformerContext): Option[(Id, List[Id], List[Id], Id, Id, Used)] = b match { - case cps.Block.BlockVar(id) => C.definitions.get(id) match { - case Some(DefInfo(id, vparams, bparams, ks, k, used)) => Some((id, vparams, bparams, ks, k, used)) - case None => None - } - case _ => None - } - } + class RecursiveUsage(var jumped: Boolean) + case class RecursiveDefInfo(id: Id, vparams: List[Id], bparams: List[Id], ks: Id, k: Id, used: RecursiveUsage) + case class ContinuationInfo(k: Id, param: Id, ks: Id) case class TransformerContext( requiresThunk: Boolean, + // known definitions of expressions (used to inline into externs) bindings: Map[Id, js.Expr], // definitions of externs (used to inline them) externs: Map[Id, cps.Extern.Def], - // currently, lexically enclosing functions and their parameters (used to determine whether a call is recursive, to rewrite into a loop) - definitions: Map[Id, DefInfo], + // the innermost (in direct style) enclosing functions (used to rewrite a definition to a loop) + recursive: Option[RecursiveDefInfo], + // the direct-style continuation, if available (used in case cps.Stmt.LetCont) + directStyle: Option[ContinuationInfo], + // the current direct-style metacontinuation + metacont: Option[Id], + // substitutions for renaming of metaconts (to avoid rebinding them) + metaconts: Map[Id, Id], // the original declaration context (used to compile pattern matching) declarations: DeclarationContext, // the usual compiler context @@ -48,18 +47,6 @@ object TransformerCps extends Transformer { ) implicit def autoContext(using C: TransformerContext): Context = C.errors - def lookup(id: Id)(using C: TransformerContext): js.Expr = C.bindings.getOrElse(id, nameRef(id)) - - def enterDefinition(id: Id, used: Used, block: cps.Block)(using C: TransformerContext): TransformerContext = block match { - case cps.BlockLit(vparams, bparams, ks, k, body) => - C.copy(definitions = Map(id -> DefInfo(id, vparams, bparams, ks, k, used))) - case _ => C - } - - def clearDefinitions(using C: TransformerContext): TransformerContext = C.copy(definitions = Map.empty) - - def bindingAll[R](bs: List[(Id, js.Expr)])(body: TransformerContext ?=> R)(using C: TransformerContext): R = - body(using C.copy(bindings = C.bindings ++ bs)) /** * Entrypoint used by the compiler to compile whole programs @@ -78,6 +65,9 @@ object TransformerCps extends Transformer { false, Map.empty, externs.collect { case d: Extern.Def => (d.id, d) }.toMap, + None, + None, + None, Map.empty, D, C) @@ -100,6 +90,9 @@ object TransformerCps extends Transformer { false, Map.empty, input.externs.collect { case d: Extern.Def => (d.id, d) }.toMap, + None, + None, + None, Map.empty, D, C) @@ -158,18 +151,18 @@ object TransformerCps extends Transformer { def toJS(id: Id, b: cps.Block)(using TransformerContext): js.Expr = b match { case cps.Block.BlockLit(vparams, bparams, ks, k, body) => - val used = new Used(false) + val used = new RecursiveUsage(false) - val translatedBody = toJS(body)(using enterDefinition(id, used, b)).stmts + val translatedBody = toJS(body)(using recursive(id, used, b)).stmts - if used.used then + if used.jumped then js.Lambda(vparams.map(nameDef) ++ bparams.map(nameDef) ++ List(nameDef(ks), nameDef(k)), List(js.While(RawExpr("true"), translatedBody, Some(uniqueName(id))))) else js.Lambda(vparams.map(nameDef) ++ bparams.map(nameDef) ++ List(nameDef(ks), nameDef(k)), translatedBody) - case other => toJS(other)(using clearDefinitions) + case other => toJS(other) } def toJS(b: cps.Block)(using TransformerContext): js.Expr = b match { @@ -185,17 +178,18 @@ object TransformerCps extends Transformer { case cps.Implementation(interface, operations) => js.Object(operations.map { case cps.Operation(id, vps, bps, ks, k, body) => - nameDef(id) -> js.Lambda(vps.map(nameDef) ++ bps.map(nameDef) ++ List(nameDef(ks), nameDef(k)), toJS(body)(using clearDefinitions).stmts) + nameDef(id) -> js.Lambda(vps.map(nameDef) ++ bps.map(nameDef) ++ List(nameDef(ks), nameDef(k)), toJS(body)(using nonrecursive(ks)).stmts) }) } - def toJS(ks: cps.MetaCont): js.Expr = nameRef(ks.id) + def toJS(ks: cps.MetaCont)(using T: TransformerContext): js.Expr = + nameRef(T.metaconts.getOrElse(ks.id, ks.id)) def toJS(k: cps.Cont)(using T: TransformerContext): js.Expr = k match { case Cont.ContVar(id) => nameRef(id) case Cont.ContLam(result, ks, body) => - js.Lambda(List(nameDef(result), nameDef(ks)), toJS(body)(using clearDefinitions).stmts) + js.Lambda(List(nameDef(result), nameDef(ks)), toJS(body)(using nonrecursive(ks)).stmts) } def toJS(e: cps.Expr)(using D: TransformerContext): js.Expr = e match { @@ -225,9 +219,18 @@ object TransformerCps extends Transformer { js.Const(nameDef(id), toJS(binding)) :: toJS(body).run(k) } - case cps.Stmt.LetCont(id, binding, body) => + // [[ let k(x, ks) = ...; if (...) jump k(42, ks2) else jump k(10, ks3) ]] = + // let x; if (...) { x = 42; ks = ks2 } else { x = 10; ks = ks3 } ... + case cps.Stmt.LetCont(id, Cont.ContLam(param, ks, body), body2) if canBeDirect(id, body2) => Binding { k => - js.Const(nameDef(id), toJS(binding)) :: requiringThunk { toJS(body)(using clearDefinitions) }.run(k) + js.Let(nameDef(param), js.Undefined) :: + toJS(body2)(using withDirectStyle(id, param, ks)).stmts ++ + toJS(body)(using directstyle(ks)).run(k) + } + + case cps.Stmt.LetCont(id, binding @ Cont.ContLam(result2, ks2, body2), body) => + Binding { k => + js.Const(nameDef(id), toJS(binding)(using nonrecursive(ks2))) :: requiringThunk { toJS(body) }.run(k) } case cps.Stmt.Match(sc, Nil, None) => @@ -245,19 +248,30 @@ object TransformerCps extends Transformer { pure(js.Switch(js.Member(scrutinee, `tag`), clauses.map { case (tag, clause) => val (e, binding) = toJS(scrutinee, tag, clause); - (e, binding.stmts) + + val stmts = binding.stmts + + stmts.last match { + case terminator : (js.Stmt.Return | js.Stmt.Break | js.Stmt.Continue) => (e, stmts) + case other => (e, stmts :+ js.Break()) + } }, default.map { s => toJS(s).stmts })) + case cps.Stmt.Jump(k, arg, ks) if D.directStyle.exists(c => c.k == k) => D.directStyle match { + case Some(ContinuationInfo(k2, param2, ks2)) => pure(js.Assign(nameRef(param2), toJS(arg))) + case None => sys error "Should not happen" + } + case cps.Stmt.Jump(k, arg, ks) => pure(js.Return(maybeThunking(js.Call(nameRef(k), toJS(arg), toJS(ks))))) - case cps.Stmt.App(callee @ DefInfo(id, vparams, bparams, ks1, k1, used), vargs, bargs, MetaCont(ks), Cont.ContVar(k)) if ks1 == ks && k1 == k => + case cps.Stmt.App(Recursive(id, vparams, bparams, ks1, k1, used), vargs, bargs, MetaCont(ks), Cont.ContVar(k)) if sameScope(ks, k, ks1, k1) => Binding { k2 => val stmts = mutable.ListBuffer.empty[js.Stmt] stmts.append(js.RawStmt("/* prepare tail call */")) - used.used = true + used.jumped = true // const x3 = [[ arg ]]; ... val vtmps = (vparams zip vargs).map { (id, arg) => @@ -336,13 +350,13 @@ object TransformerCps extends Transformer { } case cps.Stmt.Reset(prog, ks, k) => - pure(js.Return(Call(RESET, toJS(prog)(using clearDefinitions), toJS(ks), toJS(k)))) + pure(js.Return(Call(RESET, toJS(prog)(using nonrecursive(prog)), toJS(ks), toJS(k)))) case cps.Stmt.Shift(prompt, body, ks, k) => - pure(js.Return(Call(SHIFT, nameRef(prompt), noThunking { toJS(body)(using clearDefinitions) }, toJS(ks), toJS(k)))) + pure(js.Return(Call(SHIFT, nameRef(prompt), noThunking { toJS(body)(using nonrecursive(body)) }, toJS(ks), toJS(k)))) case cps.Stmt.Resume(r, b, ks2, k2) => - pure(js.Return(js.Call(RESUME, nameRef(r), toJS(b)(using clearDefinitions), toJS(ks2), toJS(k2)))) + pure(js.Return(js.Call(RESUME, nameRef(r), toJS(b)(using nonrecursive(b)), toJS(ks2), toJS(k2)))) case cps.Stmt.Hole() => pure(js.Return($effekt.call("hole"))) @@ -373,7 +387,7 @@ object TransformerCps extends Transformer { // Inlining Externs // ---------------- - def inlineExtern(id: Id, args: List[cps.Pure])(using T: TransformerContext): js.Expr = + private def inlineExtern(id: Id, args: List[cps.Pure])(using T: TransformerContext): js.Expr = T.externs.get(id) match { case Some(cps.Extern.Def(id, params, Nil, async, ExternBody.StringExternBody(featureFlag, Template(strings, templateArgs)))) if !async => @@ -383,11 +397,96 @@ object TransformerCps extends Transformer { case _ => js.Call(nameRef(id), args.map(toJS)) } - def canInline(extern: cps.Extern): Boolean = extern match { + private def canInline(extern: cps.Extern): Boolean = extern match { case cps.Extern.Def(_, _, Nil, async, ExternBody.StringExternBody(_, Template(_, _))) => !async case _ => false } + private def bindingAll[R](bs: List[(Id, js.Expr)])(body: TransformerContext ?=> R)(using C: TransformerContext): R = + body(using C.copy(bindings = C.bindings ++ bs)) + + private def lookup(id: Id)(using C: TransformerContext): js.Expr = C.bindings.getOrElse(id, nameRef(id)) + + + // Helpers for Direct-Style Transformation + // --------------------------------------- + + /** + * Used to determine whether a call with continuations [[ ks ]] (after substitution) and [[ k ]] + * is the same as the original function definition (that is [[ ks1 ]] and [[ k1 ]]. + */ + private def sameScope(ks: Id, k: Id, ks1: Id, k1: Id)(using C: TransformerContext): Boolean = + ks1 == C.metaconts.getOrElse(ks, ks) && k1 == k + + private def withDirectStyle(id: Id, param: Id, ks: Id)(using C: TransformerContext): TransformerContext = + C.copy(directStyle = Some(ContinuationInfo(id, param, ks))) + + private def recursive(id: Id, used: RecursiveUsage, block: cps.Block)(using C: TransformerContext): TransformerContext = block match { + case cps.BlockLit(vparams, bparams, ks, k, body) => + C.copy(recursive = Some(RecursiveDefInfo(id, vparams, bparams, ks, k, used)), directStyle = None, metacont = Some(ks)) + case _ => C + } + + private def nonrecursive(ks: Id)(using C: TransformerContext): TransformerContext = + C.copy(recursive = None, directStyle = None, metacont = Some(ks)) + + private def nonrecursive(block: cps.BlockLit)(using C: TransformerContext): TransformerContext = nonrecursive(block.ks) + + // ks | let k1 x1 ks1 = { let k2 x2 ks2 = jump k v ks2 }; ... = jump k v ks + private def directstyle(ks: Id)(using C: TransformerContext): TransformerContext = + val outer = C.metacont.getOrElse { sys error "Metacontinuation missing..." } + val outerSubstituted = C.metaconts.getOrElse(outer, outer) + val subst = C.metaconts.updated(ks, outerSubstituted) + C.copy(metacont = Some(ks), metaconts = subst) + + private object Recursive { + def unapply(b: cps.Block)(using C: TransformerContext): Option[(Id, List[Id], List[Id], Id, Id, RecursiveUsage)] = b match { + case cps.Block.BlockVar(id) => C.recursive.collect { + case RecursiveDefInfo(id2, vparams, bparams, ks, k, used) if id == id2 => (id, vparams, bparams, ks, k, used) + } + case _ => None + } + } + + private def canBeDirect(k: Id, stmt: Stmt)(using T: TransformerContext): Boolean = + def notIn(term: Stmt | Block | Expr | (Id, Clause) | Cont) = + val freeVars = term match { + case s: Stmt => free(s) + case b: Block => free(b) + case p: Expr => free(p) + case (id, Clause(_, body)) => free(body) + case c: Cont => free(c) + } + !freeVars.contains(k) + stmt match { + case Stmt.Jump(k2, arg, ks2) if k2 == k => notIn(arg) && T.metacont.contains(ks2.id) + case Stmt.Jump(k2, arg, ks2) => notIn(arg) + // TODO this could be a tailcall! + case Stmt.App(callee, vargs, bargs, ks, k) => notIn(stmt) + case Stmt.Invoke(callee, method, vargs, bargs, ks, k2) => notIn(stmt) + case Stmt.If(cond, thn, els) => canBeDirect(k, thn) && canBeDirect(k, els) + case Stmt.Match(scrutinee, clauses, default) => clauses.forall { + case (id, Clause(vparams, body)) => canBeDirect(k, body) + } && default.forall(body => canBeDirect(k, body)) + case Stmt.LetDef(id, binding, body) => notIn(binding) && canBeDirect(k, body) + case Stmt.LetExpr(id, binding, body) => notIn(binding) && canBeDirect(k, body) + case Stmt.LetCont(id, Cont.ContLam(result, ks2, body), body2) => + def willBeDirectItself = canBeDirect(id, body2) && canBeDirect(k, body)(using directstyle(ks2)) + def notFreeinContinuation = notIn(body) && canBeDirect(k, body2) + willBeDirectItself || notFreeinContinuation + case Stmt.Region(id, ks, body) => notIn(body) + case Stmt.Alloc(id, init, region, body) => notIn(init) && canBeDirect(k, body) + case Stmt.Var(id, init, ks2, body) => notIn(init) && canBeDirect(k, body) + case Stmt.Dealloc(ref, body) => canBeDirect(k, body) + case Stmt.Get(ref, id, body) => canBeDirect(k, body) + case Stmt.Put(ref, value, body) => notIn(value) && canBeDirect(k, body) + case Stmt.Reset(prog, ks, k) => notIn(stmt) + case Stmt.Shift(prompt, body, ks, k) => notIn(stmt) + case Stmt.Resume(resumption, body, ks, k) => notIn(stmt) + case Stmt.Hole() => true + } + + // Thunking // -------- From eb7febb36d626fe6c9149e9bd854418aba83b0f5 Mon Sep 17 00:00:00 2001 From: Jakub <168542356+JakubSchwenkbeck@users.noreply.github.com> Date: Wed, 15 Jan 2025 14:35:54 +0100 Subject: [PATCH 26/31] Add Examples / Unit Tests for Exceptions (#775) ## Motivation Resolves #480 This module demonstrates exception handling combinators in Effekt using a simple, focused example. By showcasing custom exception handling (`TestException`) with different strategies (default handling, ignoring, reporting, finalizing), it provides a practical guide for understanding these features. ## Changes - Introduced `TestException` as a custom exception type, independent of other exceptions like `OutOfBounds`. - Added `generalOperation`, which raises a `TestException` for invalid input or returns a string. - Included tests for handling exceptions with: - Default behavior - Ignoring exceptions - Reporting exceptions - Finalization hooks - "Reifying" with Results ## Testing The examples are self-contained in the `main` function, covering all provided exception-handling strategies. Each approach is tested and outputs expected results for validation (`combinators.check`). --- examples/stdlib/exception/combinators.check | 11 +++ examples/stdlib/exception/combinators.effekt | 73 ++++++++++++++++++++ 2 files changed, 84 insertions(+) create mode 100644 examples/stdlib/exception/combinators.check create mode 100644 examples/stdlib/exception/combinators.effekt diff --git a/examples/stdlib/exception/combinators.check b/examples/stdlib/exception/combinators.check new file mode 100644 index 000000000..41969b94c --- /dev/null +++ b/examples/stdlib/exception/combinators.check @@ -0,0 +1,11 @@ +Test: Default Handling +hello +hello +Error: Invalid index (0) +Error: Invalid index (-1) +hello +Test: Finalizer +hello +Test: Finalizer +Error: Invalid index (0) +Success: hello \ No newline at end of file diff --git a/examples/stdlib/exception/combinators.effekt b/examples/stdlib/exception/combinators.effekt new file mode 100644 index 000000000..cc1be1a80 --- /dev/null +++ b/examples/stdlib/exception/combinators.effekt @@ -0,0 +1,73 @@ +module examples/pos/exception/combinators + +import exception +import result + + +/// Used as a type for `Exception` purely for independent testing +record TestException() + + +/// Returns the string if index > 0; otherwise raises a TestException. +def generalOperation(str: String, index: Int): String / Exception[TestException] = { + if (index <= 0) + do raise(TestException(), "Error: Invalid index (" ++ show(index) ++ ")") + else + str +} + +def main() = { + val str: String = "hello" + + // Test for default handling of TestException + def defaultTestException { p: => String / Exception[TestException] }: Unit = { + with on[TestException].default { println("Test: Default Handling") } + println(p().show) + } + + defaultTestException { str.generalOperation(0) } // Test: Default Handling + defaultTestException { str.generalOperation(1) } // hello + + // Test for ignoring TestException + def ignoreTestException { p: => String / Exception[TestException] }: Unit = { + with on[TestException].ignore(); + println(p().show) + } + + ignoreTestException { str.generalOperation(0) } // *ignores the TestException* + ignoreTestException { str.generalOperation(1) } // hello + + // Test for reporting TestException + def reportTestException { p: => String / Exception[TestException] }: Unit = { + with on[TestException].report(); + println(p().show) + } + + reportTestException { str.generalOperation(0) } // Error: Invalid index (0) + reportTestException { str.generalOperation(-1) }// Error: Invalid index (-1) + reportTestException { str.generalOperation(1) } // hello + + // Test for finalizing TestException + def finalizeTestException { p: => String / Exception[TestException] }: Unit = { + try { + with on[TestException].finalize { println("Test: Finalizer") } + println(p().show) + } with Exception[TestException] { def raise(exception, msg) = () } + } + + finalizeTestException { str.generalOperation(0) } // Test: Finalizer + finalizeTestException { str.generalOperation(1) } // Test: Finalizer hello + + // Test for "reifying" an Exception using Result + def resultTestException { p: => String / Exception[TestException] }: Unit = { + val res = result[String, TestException] { p() } + res match { + case Success(msg) => println("Success: " ++ msg) + case Error(exc, msg) => println(msg) + } + } + + resultTestException { str.generalOperation(0) } // Error: Invalid index (0) + resultTestException { str.generalOperation(1) } // Success: hello + +} \ No newline at end of file From 1e58eeac85beeacc4617ead80d6f1313ed9e2cc7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ji=C5=99=C3=AD=20Bene=C5=A1?= Date: Wed, 15 Jan 2025 16:01:16 +0100 Subject: [PATCH 27/31] Fix #772 by parsing 'box ' instead of 'box ' (#779) Resolves #772 (if it works) The only downside now is that box-unbox inference could get some super weird and unexpected input. Oh well :) --- .../jvm/src/test/scala/effekt/RecursiveDescentTests.scala | 3 +++ effekt/shared/src/main/scala/effekt/RecursiveDescent.scala | 2 +- examples/neg/issue772.effekt | 7 +++++++ 3 files changed, 11 insertions(+), 1 deletion(-) create mode 100644 examples/neg/issue772.effekt diff --git a/effekt/jvm/src/test/scala/effekt/RecursiveDescentTests.scala b/effekt/jvm/src/test/scala/effekt/RecursiveDescentTests.scala index 7f426ffa8..5bfb2f798 100644 --- a/effekt/jvm/src/test/scala/effekt/RecursiveDescentTests.scala +++ b/effekt/jvm/src/test/scala/effekt/RecursiveDescentTests.scala @@ -140,6 +140,9 @@ class RecursiveDescentTests extends munit.FunSuite { ) parseExpr("box { (x: Int) => x }") parseExpr("box new Fresh { def fresh() = \"42\" }") + parseExpr("box foo()") + parseExpr("box bar(1)") + parseExpr("box baz(quux)") // { f } is parsed as a capture set and not backtracked. intercept[Throwable] { parseExpr("box { f }") } diff --git a/effekt/shared/src/main/scala/effekt/RecursiveDescent.scala b/effekt/shared/src/main/scala/effekt/RecursiveDescent.scala index 8dcffbeb7..65242968f 100644 --- a/effekt/shared/src/main/scala/effekt/RecursiveDescent.scala +++ b/effekt/shared/src/main/scala/effekt/RecursiveDescent.scala @@ -672,7 +672,7 @@ class RecursiveDescent(positions: Positions, tokens: Seq[Token], source: Source) val captures = `box` ~> backtrack(captureSet()) val expr = if (peek(`{`)) functionArg() else if (peek(`new`)) newExpr() - else Var(idRef()) + else callExpr() Box(captures, expr) diff --git a/examples/neg/issue772.effekt b/examples/neg/issue772.effekt new file mode 100644 index 000000000..80bc7ff93 --- /dev/null +++ b/examples/neg/issue772.effekt @@ -0,0 +1,7 @@ +def main(): Unit = { + def sayHelloTo(name: String): Unit = + println("Hello, " ++ name ++ "!") + + val _ = box sayHelloTo("Jolene") // ERROR Unbox requires a boxed type, but got Unit. + () +} From ccc3cb5f8ee303c218fd245a92cd1d1cfcba10ab Mon Sep 17 00:00:00 2001 From: Phi Date: Wed, 15 Jan 2025 17:10:23 +0100 Subject: [PATCH 28/31] Fix spelling in Computation Tour: `tyoes` -> `types` (#781) --- examples/tour/computation.effekt.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/tour/computation.effekt.md b/examples/tour/computation.effekt.md index 430f7a1e9..7bbd8259b 100644 --- a/examples/tour/computation.effekt.md +++ b/examples/tour/computation.effekt.md @@ -14,7 +14,7 @@ Examples of "computation" in Effekt are: - instances of interfaces (also known as "objects"), and - regions -Functions (and all other computation tyoes) are _second-class_ in Effekt. To make this difference explicit, we pass values in parentheses (e.g. `f(42)`) and computation in braces (e.g. `f { x => println(x) }`). +Functions (and all other computation types) are _second-class_ in Effekt. To make this difference explicit, we pass values in parentheses (e.g. `f(42)`) and computation in braces (e.g. `f { x => println(x) }`). ``` def myMap[A, B](xs: List[A]) { f: A => B }: List[B] = From 26c642df51bcdbb22186e16069d7e523d4717a85 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ji=C5=99=C3=AD=20Bene=C5=A1?= Date: Wed, 15 Jan 2025 20:52:59 +0100 Subject: [PATCH 29/31] Fix unknown compiler crash printing 'null' (#780) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Resolves the `Effect Compiler Crash: null` part of #760. The actual problem there was that `e.getMessage` can return `null` and it does for... _StackOverflowException_ πŸ₯³. Looking at the docs, it's recommended to use `e.toString`. While I was at it, I modified stack trace printing, now we do it properly by using `.printStackTrace`. This is very related to #731, I just want to get this out quickly to unblock other problems. --- .../shared/src/main/scala/effekt/Compiler.scala | 15 ++++++++------- 1 file changed, 8 insertions(+), 7 deletions(-) diff --git a/effekt/shared/src/main/scala/effekt/Compiler.scala b/effekt/shared/src/main/scala/effekt/Compiler.scala index a583f6554..c7b356e86 100644 --- a/effekt/shared/src/main/scala/effekt/Compiler.scala +++ b/effekt/shared/src/main/scala/effekt/Compiler.scala @@ -115,6 +115,11 @@ trait Compiler[Executable] { * - Server / Driver to typecheck and report type errors in VSCode */ def runFrontend(source: Source)(using C: Context): Option[Module] = + def getStackTrace(e: Throwable): String = + val stringWriter = new java.io.StringWriter() + e.printStackTrace(new java.io.PrintWriter(stringWriter)) + stringWriter.toString + try { val res = Frontend(source).map { res => val mod = res.mod @@ -128,15 +133,11 @@ trait Compiler[Executable] { None case e @ CompilerPanic(msg) => C.report(msg) - e.getStackTrace.foreach { line => - C.info(" at " + line) - } + C.info(getStackTrace(e)) None case e => - C.info("Effekt Compiler Crash: " + e.getMessage) - e.getStackTrace.foreach { line => - C.info(" at " + line) - } + C.info("Effekt Compiler Crash: " + e.toString) + C.info(getStackTrace(e)) None } From 9747e03225dd4625199979945aff2ba89ccfb936 Mon Sep 17 00:00:00 2001 From: "effekt-updater[bot]" <181701480+effekt-updater[bot]@users.noreply.github.com> Date: Mon, 20 Jan 2025 03:31:06 +0000 Subject: [PATCH 30/31] Bump version to 0.18.0 --- package.json | 2 +- pom.xml | 2 +- project/EffektVersion.scala | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/package.json b/package.json index ebb958218..7a9ed798a 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "@effekt-lang/effekt", "author": "Jonathan BrachthΓ€user", - "version": "0.17.0", + "version": "0.18.0", "repository": { "type": "git", "url": "git+https://github.com/effekt-lang/effekt.git" diff --git a/pom.xml b/pom.xml index 5aa5bb42b..9eec8c620 100644 --- a/pom.xml +++ b/pom.xml @@ -7,7 +7,7 @@ de.bstudios Effekt Effekt - 0.17.0 + 0.18.0 4.0.0 diff --git a/project/EffektVersion.scala b/project/EffektVersion.scala index 6f9aea170..e53825c5b 100644 --- a/project/EffektVersion.scala +++ b/project/EffektVersion.scala @@ -1,4 +1,4 @@ // Don't change this file without changing the CI too! import sbt.* import sbt.Keys.* -object EffektVersion { lazy val effektVersion = "0.17.0" } +object EffektVersion { lazy val effektVersion = "0.18.0" } From 719ae836718aeeae7c7842f9b07a452665ee4b96 Mon Sep 17 00:00:00 2001 From: Marvin Date: Tue, 21 Jan 2025 16:51:53 +0100 Subject: [PATCH 31/31] Fix byte-related LLVM segfaults (#785) Closes #783. And closes #778. I believe the way we currently handle size calculations for stack pushes/pops of non-structural types has some flaws for non-pointer values. Some changes in how we use the GEP instruction will probably fix this, but I find its syntax really confusing. I have a proof-of-concept alternate fix at the end. To bypass the calculation issues, we can also just change the size of a byte to 8 byte instead. This *hotfix* fixes the segfault in the original issue as well as all of my minified versions. Some additional explanations of the issue using an example program: ```scala def foo { action: Byte => Unit }: Unit = action(42.toByte) def main() = { foo { b => // does not segfault when printlns are swapped println(42) println(b) } } ```
Pseudo-flow with relevant (for the segfault) lines marked ``` main(stack) main(stack): alloc(3 * 8 = 24) push returnAddress_1 push sharer_5 push eraser_7 blockLit = vtable[blockLit_3289_clause_13] foo(blockLit, stack) foo(blockLit, stack): alloc(3 * 8 = 24) push returnAddress_40 push sharer_5 push eraser_7 literal = 42 byte = toByte(literal) vtable = blockLit[0] closure = blockLit[1] fp = *vtable // = blockLit_3289_clause_13 fp(closure, byte, stack) blockLit_3289_clause_13(closure, byte, stack): alloc(3 * 8 + 1 = 25) // <--------------------------- push byte push returnAddress_14 push sharer_26 push eraser_30 literal = 42 printlnInt(literal, stack) printlnInt(value, stack): // _4 str = show(value) unit = println(str) dealloc(3 * 8 = 24) pop _ pop _ pop returnAddress // _14 returnAddress(unit, stack) // <--------------------------- returnAddress_14(ret, stack): dealloc(1) pop byte erase(ret) alloc(3 * 8 = 24) push returnAddress_17 push sharer_5 push eraser_7 printlnByte(byte, stack) printlnByte(byte, stack): str = show(byte) unit = println(str) dealloc(24) pop _ pop _ pop returnAddress // _17 returnAddress(unit, stack) returnAddress_17(unit, stack): dealloc(24) pop _ pop _ pop returnAddress // _40 returnAddress(unit, stack) returnAddress_40(unit, stack) dealloc(24) pop _ pop _ pop returnAddress // _1 returnAddress(unit, stack) returnAddress_1(unit, stack): dealloc(24) pop _ pop _ pop returnAddress returnAddress(unit, stack) ```
The corresponding LLVM code of the relevant sections: ```llvm define tailcc void @blockLit_3289_clause_13(%Object %closure, i8 %b_2365, %Stack %stack) { entry: ; new blockLit_3289_clause_13, 1 parameters %stackPointer_33 = call ccc %StackPointer @stackAllocate(%Stack %stack, i64 25) ; <--------------- %b_2365_pointer_34 = getelementptr {i8}, %StackPointer %stackPointer_33, i64 0, i32 0 store i8 %b_2365, ptr %b_2365_pointer_34, !noalias !2 %returnAddress_pointer_35 = getelementptr {{i8}, %FrameHeader}, %StackPointer %stackPointer_33, i64 0, i32 1, i32 0 %sharer_pointer_36 = getelementptr {{i8}, %FrameHeader}, %StackPointer %stackPointer_33, i64 0, i32 1, i32 1 %eraser_pointer_37 = getelementptr {{i8}, %FrameHeader}, %StackPointer %stackPointer_33, i64 0, i32 1, i32 2 store ptr @returnAddress_14, ptr %returnAddress_pointer_35, !noalias !2 store ptr @sharer_26, ptr %sharer_pointer_36, !noalias !2 store ptr @eraser_30, ptr %eraser_pointer_37, !noalias !2 ; literalInt longLiteral_3291, n=42 %longLiteral_3291 = add i64 42, 0 ; substitution ; substitution [value_3 !-> longLiteral_3291] ; jump println_4 musttail call tailcc void @println_4(i64 %longLiteral_3291, %Stack %stack) ret void } ... define tailcc void @println_4(i64 %value_3, %Stack %stack) { entry: ; definition println_4, environment length 1 ; foreignCall pureApp_3284 : String(), foreign show_14, 1 values %pureApp_3284 = call ccc %Pos @show_14(i64 %value_3) ; foreignCall pureApp_3283 : Positive(), foreign println_1, 1 values %pureApp_3283 = call ccc %Pos @println_1(%Pos %pureApp_3284) ; substitution ; substitution [v_r_3164_3278 !-> pureApp_3283] ; return, 1 values %stackPointer_56 = call ccc %StackPointer @stackDeallocate(%Stack %stack, i64 24) %returnAddress_pointer_57 = getelementptr %FrameHeader, %StackPointer %stackPointer_56, i64 0, i32 0 %returnAddress_55 = load %ReturnAddress, ptr %returnAddress_pointer_57, !noalias !2 musttail call tailcc void %returnAddress_55(%Pos %pureApp_3283, %Stack %stack) ; <------------------------- ; here is the segfault, since returnAddress_55 is missing one byte! ret void } ``` --- A fix I came up with that does not involve making the size of a byte large: ``` llvm define tailcc void @blockLit_3289_clause_13(%Object %closure, %Byte %b_2365, %Stack %stack) { entry: ; new blockLit_3289_clause_13, 1 parameters ; SPLIT the allocation into two segments <----------------------- %stackPointar_33 = call ccc %StackPointer @stackAllocate(%Stack %stack, i64 1) %b_2365_pointer_34 = getelementptr %Byte, %StackPointer %stackPointar_33, i64 0 store %Byte %b_2365, ptr %b_2365_pointer_34, !noalias !2 %stackPointer_33 = call ccc %StackPointer @stackAllocate(%Stack %stack, i64 24) %returnAddress_pointer_35 = getelementptr %FrameHeader, %StackPointer %stackPointer_33, i64 0, i32 0 %sharer_pointer_36 = getelementptr %FrameHeader, %StackPointer %stackPointer_33, i64 0, i32 1 %eraser_pointer_37 = getelementptr %FrameHeader, %StackPointer %stackPointer_33, i64 0, i32 2 store ptr @returnAddress_14, ptr %returnAddress_pointer_35, !noalias !2 store ptr @sharer_26, ptr %sharer_pointer_36, !noalias !2 store ptr @eraser_30, ptr %eraser_pointer_37, !noalias !2 ; literalInt longLiteral_3291, n=42 %longLiteral_3291 = add i64 42, 0 ; substitution ; substitution [value_3 !-> longLiteral_3291] ; jump println_4 musttail call tailcc void @println_4(i64 %longLiteral_3291, %Stack %stack) ret void } ... ``` --------- Co-authored-by: Philipp Schuster --- effekt/jvm/src/test/scala/effekt/StdlibTests.scala | 3 --- .../main/scala/effekt/generator/llvm/Transformer.scala | 8 +------- .../src/main/scala/effekt/machine/PrettyPrinter.scala | 1 - .../src/main/scala/effekt/machine/Transformer.scala | 4 ++-- effekt/shared/src/main/scala/effekt/machine/Tree.scala | 3 ++- libraries/llvm/main.c | 1 + 6 files changed, 6 insertions(+), 14 deletions(-) diff --git a/effekt/jvm/src/test/scala/effekt/StdlibTests.scala b/effekt/jvm/src/test/scala/effekt/StdlibTests.scala index 9e6431d37..0258ee15d 100644 --- a/effekt/jvm/src/test/scala/effekt/StdlibTests.scala +++ b/effekt/jvm/src/test/scala/effekt/StdlibTests.scala @@ -41,9 +41,6 @@ class StdlibLLVMTests extends StdlibTests { override def debug = sys.env.get("EFFEKT_DEBUG").nonEmpty override def ignored: List[File] = List( - // segfaults - examplesDir / "stdlib" / "stream" / "fuse_newlines.effekt", - // Syscall param write(buf) points to uninitialised byte(s) examplesDir / "stdlib" / "io" / "filesystem" / "files.effekt", examplesDir / "stdlib" / "io" / "filesystem" / "async_file_io.effekt", diff --git a/effekt/shared/src/main/scala/effekt/generator/llvm/Transformer.scala b/effekt/shared/src/main/scala/effekt/generator/llvm/Transformer.scala index 0bd6d452e..f9e95b390 100644 --- a/effekt/shared/src/main/scala/effekt/generator/llvm/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/generator/llvm/Transformer.scala @@ -415,7 +415,6 @@ object Transformer { case machine.Type.Int() => IntegerType64() case machine.Type.Byte() => IntegerType8() case machine.Type.Double() => DoubleType() - case machine.Type.String() => positiveType case machine.Type.Reference(tpe) => referenceType } @@ -429,9 +428,8 @@ object Transformer { case machine.Type.Prompt() => 8 // TODO Make fat? case machine.Type.Stack() => 8 // TODO Make fat? case machine.Type.Int() => 8 // TODO Make fat? - case machine.Type.Byte() => 1 + case machine.Type.Byte() => 8 case machine.Type.Double() => 8 // TODO Make fat? - case machine.Type.String() => 16 case machine.Type.Reference(_) => 16 } @@ -666,7 +664,6 @@ object Transformer { case machine.Positive() => Call("_", Ccc(), VoidType(), sharePositive, List(transform(value))) case machine.Negative() => Call("_", Ccc(), VoidType(), shareNegative, List(transform(value))) case machine.Type.Stack() => Call("_", Ccc(), VoidType(), shareResumption, List(transform(value))) - case machine.Type.String() => Call("_", Ccc(), VoidType(), shareString, List(transform(value))) }.map(emit) } @@ -675,7 +672,6 @@ object Transformer { case machine.Positive() => Call("_", Ccc(), VoidType(), erasePositive, List(transform(value))) case machine.Negative() => Call("_", Ccc(), VoidType(), eraseNegative, List(transform(value))) case machine.Type.Stack() => Call("_", Ccc(), VoidType(), eraseResumption, List(transform(value))) - case machine.Type.String() => Call("_", Ccc(), VoidType(), eraseString, List(transform(value))) }.map(emit) } @@ -703,14 +699,12 @@ object Transformer { val shareNegative = ConstantGlobal("shareNegative"); val shareResumption = ConstantGlobal("shareResumption"); val shareFrames = ConstantGlobal("shareFrames"); - val shareString = ConstantGlobal("sharePositive"); val eraseObject = ConstantGlobal("eraseObject"); val erasePositive = ConstantGlobal("erasePositive"); val eraseNegative = ConstantGlobal("eraseNegative"); val eraseResumption = ConstantGlobal("eraseResumption"); val eraseFrames = ConstantGlobal("eraseFrames"); - val eraseString = ConstantGlobal("erasePositive"); val alloc = ConstantGlobal("alloc") val getPointer = ConstantGlobal("getPointer") diff --git a/effekt/shared/src/main/scala/effekt/machine/PrettyPrinter.scala b/effekt/shared/src/main/scala/effekt/machine/PrettyPrinter.scala index 00131a929..0e3e86bfa 100644 --- a/effekt/shared/src/main/scala/effekt/machine/PrettyPrinter.scala +++ b/effekt/shared/src/main/scala/effekt/machine/PrettyPrinter.scala @@ -32,7 +32,6 @@ object PrettyPrinter extends ParenPrettyPrinter { case Type.Int() => "Int" case Type.Byte() => "Byte" case Type.Double() => "Double" - case Type.String() => "String" case Type.Reference(tpe) => toDoc(tpe) <> "*" } diff --git a/effekt/shared/src/main/scala/effekt/machine/Transformer.scala b/effekt/shared/src/main/scala/effekt/machine/Transformer.scala index 1002896c9..354d13e53 100644 --- a/effekt/shared/src/main/scala/effekt/machine/Transformer.scala +++ b/effekt/shared/src/main/scala/effekt/machine/Transformer.scala @@ -390,7 +390,7 @@ object Transformer { } case core.Literal(javastring: String, _) => - val literal_binding = Variable(freshName("utf8StringLiteral"), Type.String()); + val literal_binding = Variable(freshName("utf8StringLiteral"), builtins.StringType); Binding { k => LiteralUTF8String(literal_binding, javastring.getBytes("utf-8"), k(literal_binding)) } @@ -472,7 +472,7 @@ object Transformer { case core.Type.TByte => Type.Byte() case core.Type.TBoolean => builtins.BooleanType case core.Type.TDouble => Type.Double() - case core.Type.TString => Type.String() + case core.Type.TString => Positive() case core.ValueType.Data(symbol, targs) => Positive() } diff --git a/effekt/shared/src/main/scala/effekt/machine/Tree.scala b/effekt/shared/src/main/scala/effekt/machine/Tree.scala index 8f1020665..a48f4547d 100644 --- a/effekt/shared/src/main/scala/effekt/machine/Tree.scala +++ b/effekt/shared/src/main/scala/effekt/machine/Tree.scala @@ -222,7 +222,6 @@ enum Type { case Int() case Byte() case Double() - case String() case Reference(tpe: Type) } export Type.{ Positive, Negative } @@ -241,5 +240,7 @@ object builtins { val False: Tag = 0 val BooleanType = Positive() + val StringType = Positive() + val SingletonRecord: Tag = 0 } diff --git a/libraries/llvm/main.c b/libraries/llvm/main.c index 34e22d5c9..97478d9a4 100644 --- a/libraries/llvm/main.c +++ b/libraries/llvm/main.c @@ -40,4 +40,5 @@ int main(int argc, char *argv[]) { uv_loop_t *loop = uv_default_loop(); uv_run(loop, UV_RUN_DEFAULT); uv_loop_close(loop); + return 0; }