Use injected js, trampolines for curried functions

This commit is contained in:
Dane Johnson 2024-11-14 13:09:54 -06:00
parent d0de70eb54
commit 9291e65c11
3 changed files with 28 additions and 9 deletions

View File

@ -37,6 +37,13 @@ impl <'a> LexicalContext<'a> {
}
}
/// Embeds the pre-written js
pub fn emit_injector(w: &mut dyn Write) -> std::io::Result<()>{
let bytes = include_bytes!("./js/deeinject.js");
write!(w, "{}", String::from_utf8_lossy(bytes))?;
Ok(())
}
pub fn emit_all(w: &mut dyn Write, ast: &Vec<parser::Stmt>, ctx: &mut LexicalContext) -> std::io::Result<()> {
let is_tail = ctx.is_tail;
ctx.is_tail = false;
@ -113,14 +120,17 @@ pub fn emit_expr(w: &mut dyn Write, expr: &parser::Expr, ctx: &mut LexicalContex
write!(w, "{}", atom)?;
}
parser::Expr::Funcall(id, args) => {
write!(w, "{}", munge(id))?;
if args.is_empty() {
write!(w, "()")?;
if let Some((last, butlast)) = args.split_last() {
write!(w, "T({}, ", munge(id))?;
for expr in butlast {
emit_expr(w, expr, ctx)?;
write!(w, ", ")?;
}
for arg in args {
write!(w, "(")?;
emit_expr(w, arg, ctx)?;
emit_expr(w, last, ctx)?;
write!(w, ")")?;
} else {
write!(w, "{}", munge(id))?;
write!(w, "()")?;
}
}
parser::Expr::Funcdef(arg, body) => {

10
src/js/deeinject.js Normal file
View File

@ -0,0 +1,10 @@
const print = console.log
function T(f, ...args) {
let res = f;
for (arg of args) {
res = res(arg);
}
return res;
}

View File

@ -53,8 +53,7 @@ fn script(cli: &Cli) {
println!("{:#?}", tree);
} else if cli.ecmascript {
let mut out = io::stdout();
// TODO remove this hack
writeln!(out, "const print = console.log;").ok();
emitter::emit_injector(&mut out).ok();
let mut toplevel = emitter::LexicalContext::toplevel();
emitter::emit_all(&mut out, &tree, &mut toplevel).ok();
} else {