From 7e4e936c23490eec59b9a97baa4f4bc65617a83d Mon Sep 17 00:00:00 2001 From: nezuo Date: Sat, 23 Nov 2024 17:45:07 -0800 Subject: [PATCH] Add tuples --- zap/src/config.rs | 6 +- zap/src/irgen/des.rs | 20 ++- zap/src/irgen/mod.rs | 2 +- zap/src/irgen/ser.rs | 20 ++- zap/src/output/luau/client.rs | 173 +++++++++++++++++++------- zap/src/output/luau/server.rs | 183 +++++++++++++++++++--------- zap/src/output/tooling.rs | 2 +- zap/src/output/typescript/client.rs | 12 +- zap/src/output/typescript/server.rs | 24 ++-- zap/src/parser/convert.rs | 30 ++++- zap/src/parser/grammar.lalrpop | 13 +- zap/src/parser/syntax_tree.rs | 12 +- 12 files changed, 357 insertions(+), 140 deletions(-) diff --git a/zap/src/config.rs b/zap/src/config.rs index e8e9d83e..b88383fa 100644 --- a/zap/src/config.rs +++ b/zap/src/config.rs @@ -76,8 +76,8 @@ impl std::fmt::Display for YieldType { pub struct FnDecl<'src> { pub name: &'src str, pub call: FnCall, - pub args: Option>, - pub rets: Option>, + pub args: Option>>, + pub rets: Option>>, pub id: usize, } @@ -93,7 +93,7 @@ pub struct EvDecl<'src> { pub from: EvSource, pub evty: EvType, pub call: EvCall, - pub data: Option>, + pub data: Option>>, pub id: usize, } diff --git a/zap/src/irgen/des.rs b/zap/src/irgen/des.rs index c83caad0..860ccb74 100644 --- a/zap/src/irgen/des.rs +++ b/zap/src/irgen/des.rs @@ -14,8 +14,20 @@ impl Gen for Des { self.buf.push(stmt); } - fn gen(mut self, var: Var, ty: &Ty) -> Vec { - self.push_ty(ty, var); + fn gen(mut self, var: Var, types: &[Ty]) -> Vec { + for (i, ty) in types.iter().enumerate() { + let var = if i > 0 { + Var::Name(match &var { + Var::Name(name) => format!("{name}{}", i + 1), + _ => unreachable!(), + }) + } else { + var.clone() + }; + + self.push_ty(ty, var); + } + self.buf } @@ -419,11 +431,11 @@ impl Des { } } -pub fn gen(ty: &Ty, var: &str, checks: bool) -> Vec { +pub fn gen(types: &[Ty], var: &str, checks: bool) -> Vec { Des { checks, buf: vec![], var_occurrences: HashMap::new(), } - .gen(var.into(), ty) + .gen(var.into(), types) } diff --git a/zap/src/irgen/mod.rs b/zap/src/irgen/mod.rs index 03b3c051..88c16014 100644 --- a/zap/src/irgen/mod.rs +++ b/zap/src/irgen/mod.rs @@ -9,7 +9,7 @@ pub mod ser; pub trait Gen { fn push_stmt(&mut self, stmt: Stmt); - fn gen(self, var: Var, ty: &Ty<'_>) -> Vec; + fn gen(self, var: Var, types: &[Ty]) -> Vec; fn push_local(&mut self, name: String, expr: Option) { self.push_stmt(Stmt::Local(name, expr)) diff --git a/zap/src/irgen/ser.rs b/zap/src/irgen/ser.rs index 73ca1c1a..5fcadaf0 100644 --- a/zap/src/irgen/ser.rs +++ b/zap/src/irgen/ser.rs @@ -14,8 +14,20 @@ impl Gen for Ser { self.buf.push(stmt); } - fn gen(mut self, var: Var, ty: &Ty) -> Vec { - self.push_ty(ty, var); + fn gen(mut self, var: Var, types: &[Ty]) -> Vec { + for (i, ty) in types.iter().enumerate() { + let var = if i > 0 { + Var::Name(match &var { + Var::Name(name) => format!("{name}{}", i + 1), + _ => unreachable!(), + }) + } else { + var.clone() + }; + + self.push_ty(ty, var); + } + self.buf } @@ -357,11 +369,11 @@ impl Ser { } } -pub fn gen(ty: &Ty, var: &str, checks: bool) -> Vec { +pub fn gen(types: &[Ty], var: &str, checks: bool) -> Vec { Ser { checks, buf: vec![], var_occurrences: HashMap::new(), } - .gen(var.into(), ty) + .gen(var.into(), types) } diff --git a/zap/src/output/luau/client.rs b/zap/src/output/luau/client.rs index a6703919..0e89e6ae 100644 --- a/zap/src/output/luau/client.rs +++ b/zap/src/output/luau/client.rs @@ -1,5 +1,5 @@ use crate::{ - config::{Config, EvCall, EvDecl, EvSource, EvType, FnDecl, TyDecl, YieldType}, + config::{Config, EvCall, EvDecl, EvSource, EvType, FnDecl, Ty, TyDecl, YieldType}, irgen::{des, ser}, }; @@ -103,14 +103,14 @@ impl<'src> ClientOutput<'src> { self.push_line(&format!("function types.write_{name}(value: {name})")); self.indent(); - self.push_stmts(&ser::gen(ty, "value", self.config.write_checks)); + self.push_stmts(&ser::gen(&[ty.clone()], "value", self.config.write_checks)); self.dedent(); self.push_line("end"); self.push_line(&format!("function types.read_{name}()")); self.indent(); self.push_line("local value;"); - self.push_stmts(&des::gen(ty, "value", false)); + self.push_stmts(&des::gen(&[ty.clone()], "value", false)); self.push_line("return value"); self.dedent(); self.push_line("end"); @@ -174,6 +174,23 @@ impl<'src> ClientOutput<'src> { )); } + fn get_values(&self, data: &Option>) -> String { + if let Some(types) = data { + (1..=types.len()) + .map(|i| { + if i == 1 { + "value".to_string() + } else { + format!("value{}", i) + } + }) + .collect::>() + .join(", ") + } else { + "value".to_string() + } + } + fn push_reliable_callback(&mut self, first: bool, ev: &EvDecl) { let id = ev.id; @@ -193,7 +210,9 @@ impl<'src> ClientOutput<'src> { self.indent(); - self.push_line("local value"); + let values = self.get_values(&ev.data); + + self.push_line(&format!("local {values}")); if let Some(data) = &ev.data { self.push_stmts(&des::gen(data, "value", true)); @@ -213,10 +232,10 @@ impl<'src> ClientOutput<'src> { } match ev.call { - EvCall::SingleSync => self.push_line(&format!("events[{id}](value)")), - EvCall::SingleAsync => self.push_line(&format!("task.spawn(events[{id}], value)")), - EvCall::ManySync => self.push_line("cb(value)"), - EvCall::ManyAsync => self.push_line("task.spawn(cb, value)"), + EvCall::SingleSync => self.push_line(&format!("events[{id}]({values})")), + EvCall::SingleAsync => self.push_line(&format!("task.spawn(events[{id}], {values})")), + EvCall::ManySync => self.push_line(&format!("cb({values})")), + EvCall::ManyAsync => self.push_line(&format!("task.spawn(cb, {values})")), } if ev.call == EvCall::ManySync || ev.call == EvCall::ManyAsync { @@ -228,8 +247,13 @@ impl<'src> ClientOutput<'src> { self.push_line("else"); self.indent(); - if ev.data.is_some() { - self.push_line(&format!("table.insert(event_queue[{id}], value)")); + if let Some(types) = &ev.data { + if types.len() > 1 { + self.push_line(&format!("table.insert(event_queue[{id}], {{ {values} }})")); + } else { + self.push_line(&format!("table.insert(event_queue[{id}], value)")); + } + self.push_line(&format!("if #event_queue[{id}] > 64 then")); } else { self.push_line(&format!("event_queue[{id}] += 1")); @@ -280,7 +304,9 @@ impl<'src> ClientOutput<'src> { self.push_line("local call_id = buffer.readu8(incoming_buff, read(1))"); - self.push_line("local value"); + let values = self.get_values(&fndecl.rets); + + self.push_line(&format!("local {values}")); if let Some(data) = &fndecl.rets { self.push_stmts(&des::gen(data, "value", true)); @@ -288,10 +314,10 @@ impl<'src> ClientOutput<'src> { match self.config.yield_type { YieldType::Yield | YieldType::Future => { - self.push_line(&format!("task.spawn(event_queue[{id}][call_id], value)")); + self.push_line(&format!("task.spawn(event_queue[{id}][call_id], {values})")); } YieldType::Promise => { - self.push_line(&format!("event_queue[{id}][call_id](value)")); + self.push_line(&format!("event_queue[{id}][call_id]({values})")); } } @@ -369,7 +395,9 @@ impl<'src> ClientOutput<'src> { self.indent(); - self.push_line("local value"); + let values = self.get_values(&ev.data); + + self.push_line(&format!("local {values}")); if let Some(data) = &ev.data { self.push_stmts(&des::gen(data, "value", self.config.write_checks)); @@ -389,10 +417,10 @@ impl<'src> ClientOutput<'src> { } match ev.call { - EvCall::SingleSync => self.push_line(&format!("events[{id}](value)")), - EvCall::SingleAsync => self.push_line(&format!("task.spawn(events[{id}], value)")), - EvCall::ManySync => self.push_line("cb(value)"), - EvCall::ManyAsync => self.push_line("task.spawn(cb, value)"), + EvCall::SingleSync => self.push_line(&format!("events[{id}]({values})")), + EvCall::SingleAsync => self.push_line(&format!("task.spawn(events[{id}], {values})")), + EvCall::ManySync => self.push_line(&format!("cb({values})")), + EvCall::ManyAsync => self.push_line(&format!("task.spawn(cb, {values})")), } if ev.call == EvCall::ManySync || ev.call == EvCall::ManyAsync { @@ -404,8 +432,13 @@ impl<'src> ClientOutput<'src> { self.push_line("else"); self.indent(); - if ev.data.is_some() { - self.push_line(&format!("table.insert(event_queue[{id}], value)")); + if let Some(types) = &ev.data { + if types.len() > 1 { + self.push_line(&format!("table.insert(event_queue[{id}], {{ {values} }})")); + } else { + self.push_line(&format!("table.insert(event_queue[{id}], value)")); + } + self.push_line(&format!("if #event_queue[{id}] > 64 then")); } else { self.push_line(&format!("event_queue[{id}] += 1")); @@ -515,6 +548,23 @@ impl<'src> ClientOutput<'src> { )); } + fn push_value_parameters(&mut self, types: &[Ty]) { + for (i, ty) in types.iter().enumerate() { + let value = format!( + "{}{}", + self.config.casing.with("Value", "value", "value"), + if i == 0 { "".to_string() } else { (i + 1).to_string() } + ); + + if i > 0 { + self.push(", "); + } + + self.push(&format!("{value}: ")); + self.push_ty(ty); + } + } + fn push_return_fire(&mut self, ev: &EvDecl) { let fire = self.config.casing.with("Fire", "fire", "fire"); let value = self.config.casing.with("Value", "value", "value"); @@ -522,9 +572,8 @@ impl<'src> ClientOutput<'src> { self.push_indent(); self.push(&format!("{fire} = function(")); - if let Some(data) = &ev.data { - self.push(&format!("{value}: ")); - self.push_ty(data); + if let Some(types) = &ev.data { + self.push_value_parameters(types); } self.push(")\n"); @@ -569,6 +618,14 @@ impl<'src> ClientOutput<'src> { } } + fn push_queued_value(&mut self, types: &[Ty]) { + if types.len() > 1 { + self.push("unpack(value)"); + } else { + self.push("value"); + } + } + fn push_return_setcallback(&mut self, ev: &EvDecl) { let id = ev.id; @@ -578,8 +635,8 @@ impl<'src> ClientOutput<'src> { self.push_indent(); self.push(&format!("{set_callback} = function({callback}: (")); - if let Some(data) = &ev.data { - self.push_ty(data); + if let Some(types) = &ev.data { + self.push_value_parameters(types); } self.push(") -> ()): () -> ()\n"); @@ -587,14 +644,20 @@ impl<'src> ClientOutput<'src> { self.push_line(&format!("events[{id}] = {callback}")); - if ev.data.is_some() { + if let Some(types) = &ev.data { self.push_line(&format!("for _, value in event_queue[{id}] do")); self.indent(); if ev.call == EvCall::SingleSync { - self.push_line(&format!("{callback}(value)")) + self.push_indent(); + self.push(&format!("{callback}(")); + self.push_queued_value(types); + self.push_line(")\n"); } else { - self.push_line(&format!("task.spawn({callback}, value)")) + self.push_indent(); + self.push(&format!("task.spawn({callback}, ")); + self.push_queued_value(types); + self.push(")\n"); } self.dedent(); @@ -638,8 +701,8 @@ impl<'src> ClientOutput<'src> { self.push_indent(); self.push(&format!("{on} = function({callback}: (")); - if let Some(data) = &ev.data { - self.push_ty(data); + if let Some(types) = &ev.data { + self.push_value_parameters(types); } self.push(") -> ())\n"); @@ -647,14 +710,20 @@ impl<'src> ClientOutput<'src> { self.push_line(&format!("table.insert(events[{id}], {callback})")); - if ev.data.is_some() { + if let Some(types) = &ev.data { self.push_line(&format!("for _, value in event_queue[{id}] do")); self.indent(); if ev.call == EvCall::ManySync { - self.push_line(&format!("{callback}(value)")) + self.push_indent(); + self.push(&format!("{callback}(")); + self.push_queued_value(types); + self.push_line(")\n"); } else { - self.push_line(&format!("task.spawn({callback}, value)")) + self.push_indent(); + self.push(&format!("task.spawn({callback}, ")); + self.push_queued_value(types); + self.push(")\n"); } self.dedent(); @@ -724,23 +793,35 @@ impl<'src> ClientOutput<'src> { self.push_indent(); self.push(&format!("{call} = function(")); - if let Some(ty) = &fndecl.args { - self.push(&format!("{value}: ")); - self.push_ty(ty); + if let Some(types) = &fndecl.args { + self.push_value_parameters(types); } self.push(")"); - if let Some(ty) = &fndecl.rets { + if let Some(types) = &fndecl.rets { match self.config.yield_type { YieldType::Future => { - self.push(": Future.Future<"); - self.push_ty(ty); - self.push(">"); + self.push(": Future.Future<("); + + for (i, ty) in types.iter().enumerate() { + if i > 0 { + self.push(", "); + } + self.push_ty(ty); + } + + self.push(")>"); } YieldType::Yield => { - self.push(": "); - self.push_ty(ty); + self.push(": ("); + for (i, ty) in types.iter().enumerate() { + if i > 0 { + self.push(", "); + } + self.push_ty(ty); + } + self.push(")"); } _ => (), } @@ -774,10 +855,10 @@ impl<'src> ClientOutput<'src> { match self.config.yield_type { YieldType::Yield => { self.push_line(&format!("event_queue[{id}][function_call_id] = coroutine.running()",)); - self.push_line("local async_value = coroutine.yield()"); + self.push_line("return coroutine.yield()"); } YieldType::Future => { - self.push_line("local async_value = Future.new(function()"); + self.push_line("return Future.new(function()"); self.indent(); self.push_line(&format!("event_queue[{id}][function_call_id] = coroutine.running()",)); @@ -787,7 +868,7 @@ impl<'src> ClientOutput<'src> { self.push_line("end)"); } YieldType::Promise => { - self.push_line("local async_value = Promise.new(function(resolve)"); + self.push_line("return Promise.new(function(resolve)"); self.indent(); self.push_line(&format!("event_queue[{id}][function_call_id] = resolve")); @@ -797,8 +878,6 @@ impl<'src> ClientOutput<'src> { } } - self.push_line("return async_value"); - self.dedent(); self.push_line("end,"); diff --git a/zap/src/output/luau/server.rs b/zap/src/output/luau/server.rs index 11754a78..9499a11a 100644 --- a/zap/src/output/luau/server.rs +++ b/zap/src/output/luau/server.rs @@ -1,5 +1,5 @@ use crate::{ - config::{Config, EvCall, EvDecl, EvSource, EvType, FnCall, FnDecl, TyDecl}, + config::{Config, EvCall, EvDecl, EvSource, EvType, FnCall, FnDecl, Ty, TyDecl}, irgen::{des, ser}, }; @@ -115,14 +115,14 @@ impl<'a> ServerOutput<'a> { self.push_line(&format!("function types.write_{name}(value: {name})")); self.indent(); - self.push_stmts(&ser::gen(ty, "value", self.config.write_checks)); + self.push_stmts(&ser::gen(&[ty.clone()], "value", self.config.write_checks)); self.dedent(); self.push_line("end"); self.push_line(&format!("function types.read_{name}()")); self.indent(); self.push_line("local value;"); - self.push_stmts(&des::gen(ty, "value", true)); + self.push_stmts(&des::gen(&[ty.clone()], "value", true)); self.push_line("return value"); self.dedent(); self.push_line("end"); @@ -186,6 +186,23 @@ impl<'a> ServerOutput<'a> { )); } + fn get_values(&self, data: &Option>) -> String { + if let Some(types) = data { + (1..=types.len()) + .map(|i| { + if i == 1 { + "value".to_string() + } else { + format!("value{}", i) + } + }) + .collect::>() + .join(", ") + } else { + "value".to_string() + } + } + fn push_reliable_callback(&mut self, first: bool, ev: &EvDecl) { let id = ev.id; @@ -205,7 +222,9 @@ impl<'a> ServerOutput<'a> { self.indent(); - self.push_line("local value"); + let values = self.get_values(&ev.data); + + self.push_line(&format!("local {values}")); if let Some(data) = &ev.data { self.push_stmts(&des::gen(data, "value", true)); @@ -220,10 +239,10 @@ impl<'a> ServerOutput<'a> { self.indent(); match ev.call { - EvCall::SingleSync => self.push_line(&format!("events[{id}](player, value)")), - EvCall::SingleAsync => self.push_line(&format!("task.spawn(events[{id}], player, value)")), - EvCall::ManySync => self.push_line("cb(player, value)"), - EvCall::ManyAsync => self.push_line("task.spawn(cb, player, value)"), + EvCall::SingleSync => self.push_line(&format!("events[{id}](player, {values})")), + EvCall::SingleAsync => self.push_line(&format!("task.spawn(events[{id}], player, {values})")), + EvCall::ManySync => self.push_line(&format!("cb(player, {values})")), + EvCall::ManyAsync => self.push_line(&format!("task.spawn(cb, player, {values})")), } self.dedent(); @@ -249,7 +268,10 @@ impl<'a> ServerOutput<'a> { self.indent(); self.push_line("local call_id = buffer.readu8(buff, read(1))"); - self.push_line("local value"); + + let values = self.get_values(&fndecl.args); + + self.push_line(&format!("local {values}")); if let Some(data) = &fndecl.args { self.push_stmts(&des::gen(data, "value", true)); @@ -259,12 +281,36 @@ impl<'a> ServerOutput<'a> { self.indent(); + let rets = if let Some(types) = &fndecl.args { + (1..=types.len()) + .map(|i| { + if i > 1 { + format!("rets{}", i) + } else { + "rets".to_string() + } + }) + .collect::>() + .join(", ") + } else { + "rets".to_string() + }; + if fndecl.call == FnCall::Async { + let args = if let Some(types) = &fndecl.args { + (1..=types.len()) + .map(|i| format!("value_{}", i)) + .collect::>() + .join(", ") + } else { + "value_1".to_string() + }; + // Avoid using upvalues as an optimization. - self.push_line("task.spawn(function(player_2, call_id_2, value_2)"); + self.push_line(&format!("task.spawn(function(player_2, call_id_2, {args})")); self.indent(); - self.push_line(&format!("local rets = events[{id}](player_2, value_2)")); + self.push_line(&format!("local {rets} = events[{id}](player_2, {args})")); self.push_line("load_player(player_2)"); self.push_write_event_id(fndecl.id); @@ -279,9 +325,9 @@ impl<'a> ServerOutput<'a> { self.push_line("player_map[player_2] = save()"); self.dedent(); - self.push_line("end, player, call_id, value)"); + self.push_line(&format!("end, player, call_id, {values})")); } else { - self.push_line(&format!("local rets = events[{id}](player, value)")); + self.push_line(&format!("local {rets} = events[{id}](player, {values})")); self.push_line("load_player(player)"); self.push_write_event_id(fndecl.id); @@ -371,7 +417,9 @@ impl<'a> ServerOutput<'a> { self.indent(); - self.push_line("local value"); + let values = self.get_values(&ev.data); + + self.push_line(&format!("local {}", values)); if let Some(data) = &ev.data { self.push_stmts(&des::gen(data, "value", true)); @@ -386,10 +434,10 @@ impl<'a> ServerOutput<'a> { self.indent(); match ev.call { - EvCall::SingleSync => self.push_line(&format!("events[{id}](player, value)")), - EvCall::SingleAsync => self.push_line(&format!("task.spawn(events[{id}], player, value)")), - EvCall::ManySync => self.push_line("cb(player, value)"), - EvCall::ManyAsync => self.push_line("task.spawn(cb, player, value)"), + EvCall::SingleSync => self.push_line(&format!("events[{id}](player, {values})")), + EvCall::SingleAsync => self.push_line(&format!("task.spawn(events[{id}], player, {values})")), + EvCall::ManySync => self.push_line(&format!("cb(player, {values})")), + EvCall::ManyAsync => self.push_line(&format!("task.spawn(cb, player, {values})")), } self.dedent(); @@ -447,8 +495,25 @@ impl<'a> ServerOutput<'a> { )); } + fn push_value_parameters(&mut self, types: &[Ty]) { + for (i, ty) in types.iter().enumerate() { + let value = format!( + "{}{}", + self.config.casing.with("Value", "value", "value"), + if i == 0 { "".to_string() } else { (i + 1).to_string() } + ); + + if i > 0 { + self.push(", "); + } + + self.push(&format!("{value}: ")); + self.push_ty(ty); + } + } + fn push_return_fire(&mut self, ev: &EvDecl) { - let ty = &ev.data; + let types = &ev.data; let fire = self.config.casing.with("Fire", "fire", "fire"); let player = self.config.casing.with("Player", "player", "player"); @@ -457,9 +522,9 @@ impl<'a> ServerOutput<'a> { self.push_indent(); self.push(&format!("{fire} = function({player}: Player")); - if let Some(ty) = ty { - self.push(&format!(", {value}: ")); - self.push_ty(ty); + if let Some(types) = types { + self.push(", "); + self.push_value_parameters(types); } self.push(")\n"); @@ -472,8 +537,8 @@ impl<'a> ServerOutput<'a> { self.push_write_event_id(ev.id); - if let Some(ty) = ty { - self.push_stmts(&ser::gen(ty, value, self.config.write_checks)); + if let Some(types) = types { + self.push_stmts(&ser::gen(types, value, self.config.write_checks)); } match ev.evty { @@ -490,7 +555,7 @@ impl<'a> ServerOutput<'a> { } fn push_return_fire_all(&mut self, ev: &EvDecl) { - let ty = &ev.data; + let types = &ev.data; let fire_all = self.config.casing.with("FireAll", "fireAll", "fire_all"); let value = self.config.casing.with("Value", "value", "value"); @@ -498,9 +563,8 @@ impl<'a> ServerOutput<'a> { self.push_indent(); self.push(&format!("{fire_all} = function(")); - if let Some(ty) = ty { - self.push(&format!("{value}: ")); - self.push_ty(ty); + if let Some(types) = types { + self.push_value_parameters(types); } self.push(")\n"); @@ -510,8 +574,8 @@ impl<'a> ServerOutput<'a> { self.push_write_event_id(ev.id); - if let Some(ty) = ty { - self.push_stmts(&ser::gen(ty, value, self.config.write_checks)); + if let Some(types) = types { + self.push_stmts(&ser::gen(types, value, self.config.write_checks)); } match ev.evty { @@ -540,7 +604,7 @@ impl<'a> ServerOutput<'a> { } fn push_return_fire_except(&mut self, ev: &EvDecl) { - let ty = &ev.data; + let types = &ev.data; let fire_except = self.config.casing.with("FireExcept", "fireExcept", "fire_except"); let except = self.config.casing.with("Except", "except", "except"); @@ -549,9 +613,9 @@ impl<'a> ServerOutput<'a> { self.push_indent(); self.push(&format!("{fire_except} = function({except}: Player")); - if let Some(ty) = ty { - self.push(&format!(", {value}: ")); - self.push_ty(ty); + if let Some(types) = types { + self.push(", "); + self.push_value_parameters(types); } self.push(")\n"); @@ -561,8 +625,8 @@ impl<'a> ServerOutput<'a> { self.push_write_event_id(ev.id); - if let Some(ty) = ty { - self.push_stmts(&ser::gen(ty, value, self.config.write_checks)); + if let Some(types) = types { + self.push_stmts(&ser::gen(types, value, self.config.write_checks)); } match ev.evty { @@ -603,7 +667,7 @@ impl<'a> ServerOutput<'a> { } fn push_return_fire_list(&mut self, ev: &EvDecl) { - let ty = &ev.data; + let types = &ev.data; let fire_list = self.config.casing.with("FireList", "fireList", "fire_list"); let list = self.config.casing.with("List", "list", "list"); @@ -612,9 +676,9 @@ impl<'a> ServerOutput<'a> { self.push_indent(); self.push(&format!("{fire_list} = function({list}: {{ Player }}")); - if let Some(ty) = ty { - self.push(&format!(", {value}: ")); - self.push_ty(ty); + if let Some(types) = types { + self.push(", "); + self.push_value_parameters(types); } self.push(")\n"); @@ -624,7 +688,7 @@ impl<'a> ServerOutput<'a> { self.push_write_event_id(ev.id); - if let Some(ty) = ty { + if let Some(ty) = types { self.push_stmts(&ser::gen(ty, value, self.config.write_checks)); } @@ -658,7 +722,7 @@ impl<'a> ServerOutput<'a> { } fn push_return_fire_set(&mut self, ev: &EvDecl) { - let ty = &ev.data; + let types = &ev.data; let fire_set = self.config.casing.with("FireSet", "fireSet", "fire_set"); let set = self.config.casing.with("Set", "set", "set"); @@ -667,9 +731,9 @@ impl<'a> ServerOutput<'a> { self.push_indent(); self.push(&format!("{fire_set} = function({set}: {{ [Player]: true }}")); - if let Some(ty) = ty { - self.push(&format!(", {value}: ")); - self.push_ty(ty); + if let Some(types) = types { + self.push(", "); + self.push_value_parameters(types); } self.push(")\n"); @@ -679,7 +743,7 @@ impl<'a> ServerOutput<'a> { self.push_write_event_id(ev.id); - if let Some(ty) = ty { + if let Some(ty) = types { self.push_stmts(&ser::gen(ty, value, self.config.write_checks)); } @@ -744,11 +808,11 @@ impl<'a> ServerOutput<'a> { let callback = self.config.casing.with("Callback", "callback", "callback"); self.push_indent(); - self.push(&format!("{set_callback} = function({callback}: (Player")); + self.push(&format!("{set_callback} = function({callback}: (player: Player")); - if let Some(ty) = &ev.data { + if let Some(types) = &ev.data { self.push(", "); - self.push_ty(ty); + self.push_value_parameters(types); } self.push(") -> ()): () -> ()\n"); @@ -775,11 +839,11 @@ impl<'a> ServerOutput<'a> { let callback = self.config.casing.with("Callback", "callback", "callback"); self.push_indent(); - self.push(&format!("{on} = function({callback}: (Player")); + self.push(&format!("{on} = function({callback}: (player: Player")); - if let Some(ty) = &ev.data { + if let Some(types) = &ev.data { self.push(", "); - self.push_ty(ty); + self.push_value_parameters(types); } self.push(") -> ()): () -> ()\n"); @@ -808,17 +872,22 @@ impl<'a> ServerOutput<'a> { let callback = self.config.casing.with("Callback", "callback", "callback"); self.push_indent(); - self.push(&format!("{set_callback} = function({callback}: (Player")); + self.push(&format!("{set_callback} = function({callback}: (player: Player")); - if let Some(ty) = &fndecl.args { + if let Some(types) = &fndecl.args { self.push(", "); - self.push_ty(ty); + self.push_value_parameters(types); } self.push(") -> ("); - if let Some(ty) = &fndecl.rets { - self.push_ty(ty); + if let Some(types) = &fndecl.rets { + for (i, ty) in types.iter().enumerate() { + if i > 0 { + self.push(", "); + } + self.push_ty(ty); + } } self.push(")): () -> ()\n"); diff --git a/zap/src/output/tooling.rs b/zap/src/output/tooling.rs index ebbf0a2b..6c435e3d 100644 --- a/zap/src/output/tooling.rs +++ b/zap/src/output/tooling.rs @@ -115,7 +115,7 @@ impl<'src> ToolingOutput<'src> { self.push_line(&format!("function types.read_{name}()")); self.indent(); self.push_line("local value;"); - self.push_stmts(&des::gen(ty, "value", true)); + self.push_stmts(&des::gen(&[ty.clone()], "value", true)); self.push_line("return value"); self.dedent(); self.push_line("end"); diff --git a/zap/src/output/typescript/client.rs b/zap/src/output/typescript/client.rs index bc9a4ea9..ffda867b 100644 --- a/zap/src/output/typescript/client.rs +++ b/zap/src/output/typescript/client.rs @@ -83,7 +83,8 @@ impl<'src> ClientOutput<'src> { if let Some(data) = &ev.data { self.push(value); - self.push_arg_ty(data); + todo!() + // self.push_arg_ty(data); } self.push(") => void;\n"); @@ -118,7 +119,8 @@ impl<'src> ClientOutput<'src> { if let Some(data) = &ev.data { self.push(value); - self.push_arg_ty(data); + todo!() + // self.push_arg_ty(data); } self.push(") => void) => () => void;\n"); @@ -141,7 +143,8 @@ impl<'src> ClientOutput<'src> { if let Some(data) = &fndecl.args { self.push(value); - self.push_arg_ty(data); + todo!() + // self.push_arg_ty(data); } self.push(") => "); @@ -151,7 +154,8 @@ impl<'src> ClientOutput<'src> { } if let Some(data) = &fndecl.rets { - self.push_ty(data); + todo!() + // self.push_ty(data); } else { self.push("void"); } diff --git a/zap/src/output/typescript/server.rs b/zap/src/output/typescript/server.rs index cc0b6fb4..62433f6a 100644 --- a/zap/src/output/typescript/server.rs +++ b/zap/src/output/typescript/server.rs @@ -74,7 +74,8 @@ impl<'a> ServerOutput<'a> { if let Some(data) = &ev.data { self.push(&format!(", {value}")); - self.push_arg_ty(data); + todo!() + // self.push_arg_ty(data); } self.push(") => void;\n"); @@ -89,7 +90,8 @@ impl<'a> ServerOutput<'a> { if let Some(data) = &ev.data { self.push(value); - self.push_arg_ty(data); + todo!() + // self.push_arg_ty(data); } self.push(") => void;\n"); @@ -105,7 +107,8 @@ impl<'a> ServerOutput<'a> { if let Some(data) = &ev.data { self.push(&format!(", {value}")); - self.push_arg_ty(data); + todo!() + // self.push_arg_ty(data); } self.push(") => void;\n"); @@ -121,7 +124,8 @@ impl<'a> ServerOutput<'a> { if let Some(data) = &ev.data { self.push(&format!(", {value}")); - self.push_arg_ty(data); + todo!() + // self.push_arg_ty(data); } self.push(") => void;\n"); @@ -137,7 +141,8 @@ impl<'a> ServerOutput<'a> { if let Some(data) = &ev.data { self.push(&format!(", {value}")); - self.push_arg_ty(data); + todo!() + // self.push_arg_ty(data); } self.push(") => void\n"); @@ -195,7 +200,8 @@ impl<'a> ServerOutput<'a> { if let Some(data) = &ev.data { self.push(&format!(", {value}")); - self.push_arg_ty(data); + todo!() + // self.push_arg_ty(data); } self.push(") => void) => () => void;\n"); @@ -220,13 +226,15 @@ impl<'a> ServerOutput<'a> { if let Some(data) = &fndecl.args { self.push(&format!(", {value}")); - self.push_arg_ty(data); + todo!() + // self.push_arg_ty(data); } self.push(") => "); if let Some(data) = &fndecl.rets { - self.push_ty(data); + todo!() + // self.push_ty(data); } else { self.push("void"); } diff --git a/zap/src/parser/convert.rs b/zap/src/parser/convert.rs index 54a139f6..25d9915f 100644 --- a/zap/src/parser/convert.rs +++ b/zap/src/parser/convert.rs @@ -319,10 +319,26 @@ impl<'src> Converter<'src> { let from = evdecl.from; let evty = evdecl.evty; let call = evdecl.call; - let data = evdecl.data.as_ref().map(|ty| self.ty(ty)); + let data = evdecl + .data + .as_ref() + .map(|types| types.iter().map(|ty| self.ty(ty)).collect::>()); if data.is_some() && evty == EvType::Unreliable { - let (min, max) = data.as_ref().unwrap().size(tydecls, &mut HashSet::new()); + let mut min = 0; + let mut max = Some(0); + + for ty in data.as_ref().unwrap() { + let (ty_min, ty_max) = ty.size(tydecls, &mut HashSet::new()); + + min += ty_min; + + if let (Some(ty_max), Some(max)) = (ty_max, max.as_mut()) { + *max += ty_max; + } else { + max = None; + } + } if min > self.max_unreliable_size { self.report(Report::AnalyzeOversizeUnreliable { @@ -353,8 +369,14 @@ impl<'src> Converter<'src> { fn fndecl(&mut self, fndecl: &SyntaxFnDecl<'src>, id: usize) -> FnDecl<'src> { let name = fndecl.name.name; let call = fndecl.call; - let args = fndecl.args.as_ref().map(|ty| self.ty(ty)); - let rets = fndecl.rets.as_ref().map(|ty| self.ty(ty)); + let args = fndecl + .args + .as_ref() + .map(|types| types.iter().map(|ty| self.ty(ty)).collect::>()); + let rets = fndecl + .rets + .as_ref() + .map(|types| types.iter().map(|ty| self.ty(ty)).collect::>()); FnDecl { name, diff --git a/zap/src/parser/grammar.lalrpop b/zap/src/parser/grammar.lalrpop index e98db5c2..dee0fca0 100644 --- a/zap/src/parser/grammar.lalrpop +++ b/zap/src/parser/grammar.lalrpop @@ -43,8 +43,8 @@ Decl: SyntaxDecl<'input> = { FnDecl: SyntaxFnDecl<'input> = { "funct" "=" "{" "call" ":" - )?> - )?> + )?> + )?> ","? "}" ";"? => SyntaxFnDecl { start, name, call, args, rets, end }, } @@ -58,8 +58,8 @@ EvDecl: SyntaxEvDecl<'input> = { "event" "=" "{" "from" ":" "," "type" ":" - "," "call" ":" - )?> + "," "call" ":" + )?> ","? "}" ";"? => SyntaxEvDecl { start, name, from, evty, call, data, end }, } @@ -116,6 +116,11 @@ TyKind: SyntaxTyKind<'input> = { "Instance" ")")?> => SyntaxTyKind::Instance(c), } +TyList: Vec> = { + => vec![ty], + "(" > ")" => tys, +} + Enum: SyntaxEnum<'input> = { => SyntaxEnum { start, kind, end }, } diff --git a/zap/src/parser/syntax_tree.rs b/zap/src/parser/syntax_tree.rs index 08dcce24..9acdd38c 100644 --- a/zap/src/parser/syntax_tree.rs +++ b/zap/src/parser/syntax_tree.rs @@ -78,8 +78,8 @@ pub struct SyntaxFnDecl<'src> { pub start: usize, pub name: SyntaxIdentifier<'src>, pub call: FnCall, - pub args: Option>, - pub rets: Option>, + pub args: Option>>, + pub rets: Option>>, pub end: usize, } @@ -96,7 +96,7 @@ pub struct SyntaxEvDecl<'src> { pub from: EvSource, pub evty: EvType, pub call: EvCall, - pub data: Option>, + pub data: Option>>, pub end: usize, } @@ -133,6 +133,12 @@ impl<'src> Spanned for SyntaxTy<'src> { } } +impl<'src> Spanned for Vec> { + fn span(&self) -> Span { + self.first().unwrap().start..self.last().unwrap().end + } +} + #[derive(Debug, Clone, PartialEq)] pub enum SyntaxTyKind<'src> { Num(NumTy, Option>),