use rustc_hash::FxHashSet as HashSet;
use crate::Emitter;
use crate::expressions::context::ExpressionContext;
use crate::expressions::emission::EmittedExpression;
use crate::expressions::staging::VariadicCombine;
use crate::names::go_name;
use crate::types::coercion::{Coercion, CoercionDirection};
use syntax::ast::{Annotation, Expression, UnaryOperator};
use syntax::types::Type;
struct CallArgsContext<'a> {
fn_param_types: &'a [Type],
pointer_indices: &'a HashSet<usize>,
is_go_call: bool,
is_prelude_dispatch: bool,
spread: Option<&'a Expression>,
wrap_spread_to_any: bool,
combine_variadic: Option<VariadicCombine>,
}
fn find_go_string_literal_close(s: &str) -> Option<usize> {
let bytes = s.as_bytes();
if bytes.first() != Some(&b'"') {
return None;
}
let mut i = 1;
while i < bytes.len() {
match bytes[i] {
b'\\' => i += 2,
b'"' => return Some(i),
_ => i += 1,
}
}
None
}
fn collapse_fmt_print(
function_string: &str,
args: &[Expression],
args_strings: &[String],
call_str: String,
) -> String {
if function_string != "fmt.Print" && function_string != "fmt.Println" {
return call_str;
}
if args_strings.len() != 1 {
return call_str;
}
let arg = &args_strings[0];
if let Some(inner) = arg
.strip_prefix("fmt.Sprintf(")
.and_then(|s| s.strip_suffix(')'))
{
let suffix = if function_string == "fmt.Println" {
"\\n"
} else {
""
};
if suffix.is_empty() {
return format!("fmt.Printf({})", inner);
}
if let Some(close_quote) = find_go_string_literal_close(inner) {
let format_open = &inner[..close_quote];
let close_and_rest = &inner[close_quote..];
return format!("fmt.Printf({}{}{})", format_open, suffix, close_and_rest);
}
return call_str;
}
if let Some(arg_expression) = args.first()
&& let Expression::Call {
expression: inner_callee,
args: inner_args,
spread,
..
} = arg_expression.unwrap_parens()
&& spread.is_none()
&& inner_args.len() == 1
&& inner_callee.unwrap_parens().as_dotted_path().as_deref() == Some("fmt.Sprint")
&& let Some(inner) = arg
.strip_prefix("fmt.Sprint(")
.and_then(|s| s.strip_suffix(')'))
{
return format!("{}({})", function_string, inner);
}
call_str
}
impl Emitter<'_> {
pub(super) fn emit_regular_call(
&mut self,
output: &mut String,
call_expression: &Expression,
call_ty: Option<&Type>,
expression_ctx: ExpressionContext<'_>,
) -> String {
let Expression::Call {
expression: callee,
args,
type_args,
spread,
..
} = call_expression
else {
unreachable!("emit_regular_call requires a Call expression");
};
let function = callee.unwrap_parens();
let spread = (**spread).as_ref();
if let Some(go_name) = self.get_callee_go_name(function).map(str::to_string) {
let stages: Vec<EmittedExpression> = args
.iter()
.map(|a| self.stage_operand(a, ExpressionContext::value()))
.collect();
let wrap_to_any = Self::spread_needs_any_wrap(function, spread);
let combine = Self::variadic_combine_for(function, spread, 0);
let args_strings =
self.sequence_with_spread(output, stages, spread, wrap_to_any, "_arg", combine);
return format!("{}({})", go_name, args_strings.join(", "));
}
let mut function_string = self.emit_operand(output, function, expression_ctx.callee());
if matches!(
function,
Expression::Unary {
operator: UnaryOperator::Deref,
..
}
) {
function_string = format!("({})", function_string);
}
let type_args_string = self.resolve_call_type_args(
function,
type_args,
call_ty,
&mut function_string,
expression_ctx,
);
let pointer_indices = self.get_recursive_enum_pointer_indices(function);
let fn_param_types: Vec<Type> = match function.get_type().unwrap_forall() {
Type::Function { params, .. } => params.clone(),
_ => vec![],
};
let (is_go_call, is_prelude_dispatch) = match function.unwrap_parens() {
Expression::DotAccess { expression, .. } => {
let is_prelude = matches!(
expression.get_type().strip_refs().unwrap_forall(),
Type::Nominal { id, .. } if id.starts_with("prelude.")
);
(Self::is_go_receiver(expression), is_prelude)
}
Expression::Identifier {
qualified: Some(q), ..
} if q.starts_with("prelude.") => (false, true),
_ => (false, false),
};
let args_ctx = CallArgsContext {
fn_param_types: &fn_param_types,
pointer_indices: &pointer_indices,
is_go_call,
is_prelude_dispatch,
spread,
wrap_spread_to_any: Self::spread_needs_any_wrap(function, spread),
combine_variadic: Self::variadic_combine_for(function, spread, 0),
};
let args_strings = self.emit_call_args(output, args, &args_ctx);
let call_str = format!(
"{}{}({})",
function_string,
type_args_string,
args_strings.join(", ")
);
let call_str = collapse_fmt_print(&function_string, args, &args_strings, call_str);
if let Some(wrapped) =
self.wrap_go_array_return(output, function, &call_str, expression_ctx)
{
return wrapped;
}
call_str
}
fn wrap_go_array_return(
&mut self,
output: &mut String,
function: &Expression,
call_str: &str,
ctx: ExpressionContext<'_>,
) -> Option<String> {
if ctx.keeps_raw_go_array_return() {
return None;
}
let Expression::DotAccess {
expression: receiver_expression,
member,
..
} = function.unwrap_parens()
else {
return None;
};
if !Self::is_go_receiver(receiver_expression)
|| !self.has_go_array_return(receiver_expression, member)
{
return None;
}
let temp = self.hoist_tmp_value(output, "arr", call_str);
Some(format!("{}[:]", temp))
}
fn resolve_call_type_args(
&mut self,
function: &Expression,
type_args: &[Annotation],
call_ty: Option<&Type>,
function_string: &mut String,
ctx: ExpressionContext<'_>,
) -> String {
let mut type_args_string = self.format_type_args_from_annotations(type_args);
let slot_ty = ctx.expected_slot_type();
if type_args_string.is_empty()
&& let Some(inferred) = self.infer_return_only_type_args(function)
{
type_args_string = slot_ty
.and_then(|t| self.prelude_container_type_args(t))
.unwrap_or(inferred);
}
if type_args_string.is_empty() && Self::is_prelude_variant_constructor(function) {
let candidate = call_ty
.and_then(|t| self.prelude_container_type_args(t))
.or_else(|| slot_ty.and_then(|t| self.prelude_container_type_args(t)));
type_args_string = candidate.unwrap_or_default();
}
if !type_args_string.is_empty()
&& let Some(bracket_start) = function_string.find('[')
{
function_string.truncate(bracket_start);
}
type_args_string
}
fn emit_call_args(
&mut self,
output: &mut String,
args: &[Expression],
ctx: &CallArgsContext<'_>,
) -> Vec<String> {
let stages: Vec<EmittedExpression> = args
.iter()
.enumerate()
.map(|(i, arg)| {
let mut setup = String::new();
let value = self.emit_call_arg(&mut setup, arg, i, ctx);
EmittedExpression::new(setup, value, arg)
})
.collect();
self.sequence_with_spread(
output,
stages,
ctx.spread,
ctx.wrap_spread_to_any,
"_arg",
ctx.combine_variadic.as_ref().cloned(),
)
}
pub(crate) fn variadic_combine_for(
function: &Expression,
spread: Option<&Expression>,
extra_leading: usize,
) -> Option<VariadicCombine> {
spread?;
let fn_ty = function.get_type();
let unwrapped = fn_ty.unwrap_forall();
let elem_ty = unwrapped.is_variadic()?;
let fixed_in_signature = unwrapped.get_function_params()?.len().saturating_sub(1);
Some(VariadicCombine {
elem_ty,
fixed_count: fixed_in_signature + extra_leading,
})
}
fn spread_needs_any_wrap(function: &Expression, spread: Option<&Expression>) -> bool {
let Some(spread_expr) = spread else {
return false;
};
let Some(variadic_elem) = function.get_type().unwrap_forall().is_variadic() else {
return false;
};
if !variadic_elem.is_unknown() {
return false;
}
spread_expr
.get_type()
.inner()
.is_some_and(|t| !t.is_unknown())
}
fn emit_call_arg(
&mut self,
output: &mut String,
arg: &Expression,
index: usize,
ctx: &CallArgsContext<'_>,
) -> String {
let effective_param_ty = self.effective_param_type(index, ctx.fn_param_types);
if ctx.is_go_call
&& let Some(result) = self.try_emit_callback_wrapper(output, arg, effective_param_ty)
{
return result;
}
if let Some(result) = self.try_emit_nullable_coercion(output, arg, effective_param_ty) {
return result;
}
if ctx.is_go_call
&& let Some(result) =
self.try_emit_go_pointer_param_unwrap(output, arg, effective_param_ty)
{
return result;
}
if ctx.pointer_indices.contains(&index) {
let value = self.emit_value(output, arg, ExpressionContext::value());
if matches!(arg, Expression::Reference { .. }) || arg.get_type().is_ref() {
return value;
}
let temp = self.hoist_tmp_value(output, "ptr", &value);
return format!("&{}", temp);
}
let unwrapped_param_ty = effective_param_ty.map(|p| p.unwrap_forall());
let suppress = ctx.is_prelude_dispatch
&& unwrapped_param_ty.is_some_and(|p| matches!(p, Type::Function { .. }));
let flows_to_unknown = unwrapped_param_ty.is_some_and(|p| p.resolves_to_unknown());
let arg_ctx = ExpressionContext::value()
.with_forced_tagged_go_function(suppress)
.with_unknown_argument_target(flows_to_unknown);
let value = self.emit_composite_value(output, arg, arg_ctx);
if suppress
&& let Some(tagged) =
self.try_lower_arg_to_tagged(output, arg, &value, effective_param_ty)
{
return tagged;
}
match effective_param_ty {
Some(target) => {
let coercion =
Coercion::resolve(self, &arg.get_type(), target, CoercionDirection::Internal);
coercion.apply(self, output, value)
}
None => value,
}
}
fn effective_param_type<'a>(
&self,
index: usize,
fn_param_types: &'a [Type],
) -> Option<&'a Type> {
fn_param_types.get(index).or_else(|| {
fn_param_types
.last()
.filter(|t| t.get_name() == Some("VarArgs"))
})
}
fn try_emit_callback_wrapper(
&mut self,
output: &mut String,
arg: &Expression,
effective_param_ty: Option<&Type>,
) -> Option<String> {
let param_fn_ty = effective_param_ty
.and_then(|param_ty| {
self.facts
.resolve_to_function_type(param_ty.unwrap_forall())
})
.filter(|fn_ty| {
let Type::Function { return_type, .. } = fn_ty else {
return false;
};
return_type.is_result()
|| return_type.is_option()
|| return_type.tuple_arity().is_some_and(|a| a >= 2)
})?;
let arg_ty = arg.get_type();
let arg_fn_ty = self.facts.resolve_to_function_type(arg_ty.unwrap_forall());
if let Some(Type::Function {
return_type: arg_ret,
..
}) = arg_fn_ty.as_ref()
&& let Type::Function {
return_type: param_ret,
..
} = ¶m_fn_ty
&& self.classify_direct_emission(arg_ret).is_some()
&& self.classify_direct_emission(param_ret).is_some()
{
return Some(self.emit_value(output, arg, ExpressionContext::value()));
}
let value = self.emit_value(output, arg, ExpressionContext::value());
Some(crate::types::abi_transition::emit_lisette_callback_wrapper(
self,
output,
&value,
¶m_fn_ty,
))
}
fn try_emit_go_pointer_param_unwrap(
&mut self,
output: &mut String,
arg: &Expression,
effective_param_ty: Option<&Type>,
) -> Option<String> {
let param_ty = effective_param_ty?;
let arg_ty = arg.get_type();
let shapes_match = (self.facts.is_nullable_option(param_ty)
&& self.facts.is_nullable_option(&arg_ty))
|| (self.is_non_nilable_option(param_ty) && self.is_non_nilable_option(&arg_ty));
if !shapes_match {
return None;
}
if matches!(arg, Expression::Identifier { value, .. } if value == "None") {
return Some("nil".to_string());
}
let value = self.emit_value(output, arg, ExpressionContext::value());
let coercion = Coercion::resolve(self, &arg_ty, param_ty, CoercionDirection::ToGoBoundary);
Some(coercion.apply(self, output, value))
}
fn try_emit_nullable_coercion(
&mut self,
output: &mut String,
arg: &Expression,
effective_param_ty: Option<&Type>,
) -> Option<String> {
let param_ty = effective_param_ty?;
let arg_ty = arg.get_type();
if !self.facts.is_nullable_option(&arg_ty) {
return None;
}
let check_ty = if param_ty.get_name() == Some("VarArgs") {
param_ty.inner().unwrap_or_else(|| param_ty.clone())
} else {
param_ty.clone()
};
let needs_coercion = self
.facts
.as_interface(&check_ty)
.is_some_and(|id| go_name::is_go_import(&id))
|| (check_ty.has_name("Unknown") && {
let inner = arg_ty.ok_type();
self.facts
.as_interface(&inner)
.is_some_and(|id| go_name::is_go_import(&id))
});
if !needs_coercion {
return None;
}
Some(self.emit_unwrap_go_nullable_arg(output, arg, &arg_ty))
}
fn emit_unwrap_go_nullable_arg(
&mut self,
output: &mut String,
arg: &Expression,
arg_ty: &Type,
) -> String {
if matches!(arg, Expression::Identifier { value, .. } if value == "None") {
return "nil".to_string();
}
let value = self.emit_value(output, arg, ExpressionContext::value());
let coercion = Coercion::resolve(self, arg_ty, arg_ty, CoercionDirection::ToGoBoundary);
coercion.apply(self, output, value)
}
}