use crate::Emitter;
use crate::control_flow::fallible::{ConstructorKind, Fallible, FallibleEmitter};
use crate::types::emitter::Position;
use crate::utils::{inline_trivial_bindings, optimize_region};
use crate::write_line;
use syntax::ast::Expression;
use syntax::types::Type;
impl Emitter<'_> {
pub(crate) fn emit_propagate(
&mut self,
output: &mut String,
expression: &Expression,
result_var_name: Option<&str>,
) -> String {
let expression_ty = expression.get_type().resolve();
let fallible = Fallible::from_type(&expression_ty)
.expect("emit_propagate called on non-Result/Option type");
if let Some(var_name) = result_var_name
&& let Some(result) = self.try_emit_error_constructor(output, expression, &fallible)
{
if var_name != "_" {
let inner_ty = fallible.ok_ty();
let zero = self.zero_value(inner_ty);
let go_ty = self.go_type_as_string(inner_ty);
write_line!(output, "var {} {} = {}", var_name, go_ty, zero);
self.declare(var_name);
}
return result;
}
self.flags.needs_stdlib = true;
let check_var = if let Expression::Identifier { value, ty, .. } = expression {
let go_name = self.emit_identifier(value, ty);
if go_name.contains('(') {
self.capture_check_var(output, &go_name)
} else {
go_name
}
} else {
let expression_string = self.emit_operand(output, expression);
self.capture_check_var(output, &expression_string)
};
let result_var = result_var_name.map(|s| s.to_string()).unwrap_or_else(|| {
let v = self.fresh_var(Some("result"));
self.declare(&v);
v
});
let err_field = if fallible.is_result() { ".ErrVal" } else { "" };
let err_return = {
let mut fe = FallibleEmitter::new(self, &fallible);
fe.emit_contextual_failure(Some(&format!("{}{}", check_var, err_field)))
};
write_line!(
output,
"if {}.Tag != {} {{\nreturn {}\n}}",
check_var,
fallible.success_tag(),
err_return
);
if result_var != "_" {
write_line!(
output,
"{} := {}.{}",
result_var,
check_var,
fallible.ok_field()
);
}
result_var
}
fn capture_check_var(&mut self, output: &mut String, expression_string: &str) -> String {
let check_var = self.fresh_var(Some("check"));
self.declare(&check_var);
write_line!(output, "{} := {}", check_var, expression_string);
check_var
}
pub(crate) fn emit_option_result_assignment(
&mut self,
output: &mut String,
target_var: &str,
expression: &Expression,
) {
let target_ty = self.assign_target_ty.as_ref().map(|t| t.resolve());
let ty = target_ty
.filter(|t| t.is_option() || t.is_result())
.unwrap_or_else(|| expression.get_type());
let Some(fallible) = Fallible::from_type(&ty) else {
let expression_string = self.emit_operand(output, expression);
write_line!(output, "{} = {}", target_var, expression_string);
return;
};
let actual_expression = if let Expression::Block { items, .. } = expression {
if items.len() == 1 {
&items[0]
} else {
expression
}
} else {
expression
};
match actual_expression {
Expression::Call {
expression: callee,
args,
..
} => {
let kind = fallible.classify_constructor(callee);
let constructor_name = match kind {
Some(ConstructorKind::Success) => fallible.ok_constructor(),
Some(ConstructorKind::Failure) => fallible.err_constructor(),
None => {
let expression_string = self.emit_operand(output, expression);
write_line!(output, "{} = {}", target_var, expression_string);
return;
}
};
let mut fe = FallibleEmitter::new(self, &fallible);
if kind == Some(ConstructorKind::Success)
|| (kind == Some(ConstructorKind::Failure)
&& fallible.err_constructor_takes_arg())
{
let arg = fe.emitter.emit_composite_value(output, &args[0]);
let call_str = fe.format_constructor_call(constructor_name, Some(&arg));
write_line!(output, "{} = {}", target_var, call_str);
} else {
let call_str = fe.format_constructor_call(constructor_name, None);
write_line!(output, "{} = {}", target_var, call_str);
}
}
Expression::Identifier { .. } => {
if fallible.classify_constructor(actual_expression)
== Some(ConstructorKind::Failure)
{
let mut fe = FallibleEmitter::new(self, &fallible);
let call_str = fe.format_constructor_call(fallible.err_constructor(), None);
write_line!(output, "{} = {}", target_var, call_str);
} else {
let expression_string = self.emit_operand(output, expression);
write_line!(output, "{} = {}", target_var, expression_string);
}
}
_ => {
self.emit_block_to_var_with_braces(output, expression, target_var, false);
}
}
}
pub(crate) fn emit_propagate_to_let(
&mut self,
output: &mut String,
var_name: &str,
expression: &Expression,
) {
let Expression::Propagate { expression, .. } = expression else {
return;
};
self.emit_propagate(output, expression, Some(var_name));
}
pub(crate) fn emit_return(&mut self, output: &mut String, expression: &Expression) {
let is_unit = self
.current_return_context
.as_ref()
.is_some_and(|ty| ty.is_unit());
if is_unit {
let is_pure = matches!(
expression,
Expression::Unit { .. }
| Expression::Identifier { .. }
| Expression::Literal { .. }
);
if !is_pure {
self.emit_statement(output, expression);
}
output.push_str("return\n");
} else if !self.emit_wrapped_return(output, expression) {
let expression_string =
self.with_position(Position::Tail, |this| this.emit_value(output, expression));
let expression_string = self.adapt_return_to_context(expression, expression_string);
write_line!(output, "return {}", expression_string);
}
}
pub(crate) fn adapt_return_to_context(
&mut self,
expression: &Expression,
emitted: String,
) -> String {
let Some(return_ty) = self.current_return_context.clone() else {
return emitted;
};
self.maybe_wrap_as_go_interface(emitted, &expression.get_type(), &return_ty)
}
pub(crate) fn emit_wrapped_return(
&mut self,
output: &mut String,
expression: &Expression,
) -> bool {
let expression_ty = expression.get_type().resolve();
let return_ty = self
.current_return_context
.as_ref()
.map(|ty| ty.resolve())
.filter(|ctx_ty| Fallible::from_type(ctx_ty).is_some())
.unwrap_or(expression_ty);
let Some(fallible) = Fallible::from_type(&return_ty) else {
return false;
};
self.flags.needs_stdlib = true;
if let Expression::Identifier { .. } = expression
&& fallible.classify_constructor(expression) == Some(ConstructorKind::Failure)
{
let mut fe = FallibleEmitter::new(self, &fallible);
let failure = fe.emit_failure(None);
write_line!(output, "return {}", failure);
return true;
}
if matches!(expression, Expression::Call { .. }) {
self.emit_wrapped_call_return(output, expression, &fallible, &return_ty);
return true;
}
if matches!(expression, Expression::If { .. } | Expression::Match { .. }) {
self.emit_wrapped_branching_return(output, expression, &fallible, &return_ty);
return true;
}
let value = self.emit_value(output, expression);
write_line!(output, "return {}", value);
true
}
fn emit_wrapped_call_return(
&mut self,
output: &mut String,
expression: &Expression,
fallible: &Fallible,
return_ty: &Type,
) {
let Expression::Call {
expression: call_expression,
args,
..
} = expression
else {
unreachable!("emit_wrapped_call_return requires a Call expression");
};
match fallible.classify_constructor(call_expression) {
Some(ConstructorKind::Success) => {
let arg = self.emit_composite_value(output, &args[0]);
let mut fe = FallibleEmitter::new(self, fallible);
let success = fe.emit_success(&arg);
write_line!(output, "return {}", success);
}
Some(ConstructorKind::Failure) => {
let failure = if fallible.is_result() {
let arg = self.emit_composite_value(output, &args[0]);
let mut fe = FallibleEmitter::new(self, fallible);
fe.emit_failure(Some(&arg))
} else {
let mut fe = FallibleEmitter::new(self, fallible);
fe.emit_failure(None)
};
write_line!(output, "return {}", failure);
}
None => {
if let Some(strategy) = self.resolve_go_call_strategy(expression) {
let result_var =
self.emit_go_wrapped_call(output, expression, &strategy, return_ty);
write_line!(output, "return {}", result_var);
} else {
let call = self.emit_call(output, expression, None);
write_line!(output, "return {}", call);
}
}
}
}
fn emit_wrapped_branching_return(
&mut self,
output: &mut String,
expression: &Expression,
fallible: &Fallible,
return_ty: &Type,
) {
let temp_var = self.fresh_var(None);
self.declare(&temp_var);
let full_ty = {
let mut fe = FallibleEmitter::new(self, fallible);
fe.full_type_string()
};
let pre_len = output.len();
write_line!(output, "var {} {}", temp_var, full_ty);
let saved_target_ty = self.assign_target_ty.replace(return_ty.clone());
self.with_position(Position::Assign(temp_var.clone()), |this| {
this.emit_branching_directly(output, expression);
});
self.assign_target_ty = saved_target_ty;
write_line!(output, "return {}", temp_var);
optimize_region(output, pre_len, Some(&temp_var));
}
pub(crate) fn emit_try_block(
&mut self,
output: &mut String,
items: &[Expression],
ty: &Type,
) -> String {
self.flags.needs_stdlib = true;
let effective_ty = self.resolve_fallible_block_type(items, ty);
let fallible = Fallible::from_type(&effective_ty)
.expect("`try` block must have Result or Option type");
let result_var = self.fresh_var(Some("tryResult"));
self.declare(&result_var);
let full_ty = {
let mut fe = FallibleEmitter::new(self, &fallible);
fe.full_type_string()
};
write_line!(output, "{} := func() {} {{", result_var, full_ty);
let closure_body_start = output.len();
let saved_return_context = self.current_return_context.clone();
self.current_return_context = Some(effective_ty.clone());
self.with_fresh_scope(|emitter| {
emitter.emit_try_body(output, items, &fallible);
});
self.current_return_context = saved_return_context;
inline_trivial_bindings(output, closure_body_start);
output.push_str("}()\n");
result_var
}
fn resolve_fallible_block_type(&self, items: &[Expression], ty: &Type) -> Type {
let tail_is_never = items.last().is_some_and(|last| {
let t = last.get_type().resolve();
t.is_never() || last.diverges().is_some()
});
let base = Fallible::from_type(ty);
let needs_return_context = tail_is_never
|| base
.as_ref()
.is_some_and(|f| f.ok_ty().is_variable() || f.ok_ty().is_never());
if !needs_return_context {
return ty.clone();
}
self.current_return_context
.as_ref()
.filter(|ctx_ty| Fallible::from_type(ctx_ty).is_some())
.cloned()
.unwrap_or_else(|| ty.clone())
}
fn emit_try_body(&mut self, output: &mut String, items: &[Expression], fallible: &Fallible) {
let Some((last, rest)) = items.split_last() else {
self.emit_try_unit_return(output, fallible);
return;
};
for item in rest {
self.emit_statement(output, item);
}
self.emit_try_tail(output, last, fallible);
}
fn emit_try_tail(&mut self, output: &mut String, last: &Expression, fallible: &Fallible) {
if last.diverges().is_some() || last.get_type().resolve().is_never() {
self.emit_statement(output, last);
if !Self::is_go_never(last) {
output.push_str("panic(\"unreachable\")\n");
}
return;
}
let is_statement_only = matches!(
last,
Expression::Let { .. }
| Expression::Const { .. }
| Expression::Assignment { .. }
| Expression::While { .. }
| Expression::WhileLet { .. }
| Expression::For { .. }
| Expression::Loop { .. }
);
let is_unit_call = last.get_type().resolve().is_unit()
&& matches!(last.unwrap_parens(), Expression::Call { .. });
if is_statement_only || is_unit_call {
self.emit_statement(output, last);
self.emit_try_unit_return(output, fallible);
return;
}
let final_expression = self.emit_value(output, last);
if final_expression.is_empty() {
self.emit_try_unit_return(output, fallible);
} else {
self.emit_try_success_return(output, &final_expression, fallible);
}
}
fn emit_try_unit_return(&mut self, output: &mut String, fallible: &Fallible) {
let unit_val = self.zero_value(fallible.ok_ty());
self.emit_try_success_return(output, &unit_val, fallible);
}
fn emit_try_success_return(&mut self, output: &mut String, value: &str, fallible: &Fallible) {
let ok_return = {
let mut fe = FallibleEmitter::new(self, fallible);
fe.emit_success(value)
};
write_line!(output, "return {}", ok_return);
}
fn try_emit_error_constructor(
&mut self,
output: &mut String,
expression: &Expression,
fallible: &Fallible,
) -> Option<String> {
let err_arg = match expression {
Expression::Call {
expression: func,
args,
..
} => {
if fallible.classify_constructor(func) != Some(ConstructorKind::Failure) {
return None;
}
if !args.is_empty() {
Some(self.emit_value(output, &args[0]))
} else {
Some(String::new())
}
}
Expression::Identifier { .. } => {
if fallible.classify_constructor(expression) != Some(ConstructorKind::Failure) {
return None;
}
Some(String::new())
}
_ => return None,
};
self.flags.needs_stdlib = true;
let err_return = {
let mut fe = FallibleEmitter::new(self, fallible);
fe.emit_contextual_failure(err_arg.as_deref())
};
write_line!(output, "return {}", err_return);
Some(String::new())
}
pub(crate) fn emit_recover_block(
&mut self,
output: &mut String,
items: &[Expression],
ty: &Type,
) -> String {
self.flags.needs_stdlib = true;
let effective_ty = self.resolve_fallible_block_type(items, ty);
let fallible = Fallible::from_type(&effective_ty)
.expect("recover block type must be Result<T, PanicValue>");
let result_var = self.fresh_var(Some("recoverResult"));
self.declare(&result_var);
let inner_ty_str = self.go_type_as_string(fallible.ok_ty());
write_line!(
output,
"{} := lisette.RecoverBlock(func() {} {{",
result_var,
inner_ty_str
);
let saved_return_context = self.current_return_context.clone();
self.current_return_context = Some(fallible.ok_ty().clone());
self.with_fresh_scope(|emitter| {
emitter.emit_recover_body(output, items, &fallible);
});
self.current_return_context = saved_return_context;
output.push_str("})\n");
result_var
}
fn emit_recover_body(
&mut self,
output: &mut String,
items: &[Expression],
fallible: &Fallible,
) {
let Some((last, rest)) = items.split_last() else {
let zero = self.zero_value(fallible.ok_ty());
write_line!(output, "return {}", zero);
return;
};
for item in rest {
self.emit_statement(output, item);
}
self.emit_recover_tail(output, last, fallible);
}
fn emit_recover_tail(&mut self, output: &mut String, last: &Expression, fallible: &Fallible) {
let item_ty = last.get_type().resolve();
if item_ty.is_never() {
self.emit_statement(output, last);
if !Self::is_go_never(last) {
output.push_str("panic(\"unreachable\")\n");
}
return;
}
if item_ty.is_unit() || item_ty.is_variable() {
self.emit_statement(output, last);
let zero = self.zero_value(fallible.ok_ty());
write_line!(output, "return {}", zero);
return;
}
let expression = self.emit_value(output, last);
write_line!(output, "return {}", expression);
}
}