mirror of
https://github.com/roc-lang/roc.git
synced 2025-09-28 06:14:46 +00:00
List.contains in Zig
This commit is contained in:
parent
a3d0dc2fc3
commit
7bd228ca9b
5 changed files with 102 additions and 199 deletions
|
@ -65,3 +65,4 @@ pub const LIST_KEEP_OKS: &str = "roc_builtins.list.keep_oks";
|
||||||
pub const LIST_KEEP_ERRS: &str = "roc_builtins.list.keep_errs";
|
pub const LIST_KEEP_ERRS: &str = "roc_builtins.list.keep_errs";
|
||||||
pub const LIST_WALK: &str = "roc_builtins.list.walk";
|
pub const LIST_WALK: &str = "roc_builtins.list.walk";
|
||||||
pub const LIST_WALK_BACKWARDS: &str = "roc_builtins.list.walk_backwards";
|
pub const LIST_WALK_BACKWARDS: &str = "roc_builtins.list.walk_backwards";
|
||||||
|
pub const LIST_CONTAINS: &str = "roc_builtins.list.contains";
|
||||||
|
|
|
@ -300,3 +300,77 @@ pub fn build_rc_wrapper<'a, 'ctx, 'env>(
|
||||||
|
|
||||||
function_value
|
function_value
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn build_eq_wrapper<'a, 'ctx, 'env>(
|
||||||
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
|
layout_ids: &mut LayoutIds<'a>,
|
||||||
|
layout: &Layout<'a>,
|
||||||
|
) -> FunctionValue<'ctx> {
|
||||||
|
let block = env.builder.get_insert_block().expect("to be in a function");
|
||||||
|
let di_location = env.builder.get_current_debug_location().unwrap();
|
||||||
|
|
||||||
|
let symbol = Symbol::GENERIC_EQ_REF;
|
||||||
|
let fn_name = layout_ids
|
||||||
|
.get(symbol, &layout)
|
||||||
|
.to_symbol_string(symbol, &env.interns);
|
||||||
|
|
||||||
|
let function_value = match env.module.get_function(fn_name.as_str()) {
|
||||||
|
Some(function_value) => function_value,
|
||||||
|
None => {
|
||||||
|
let arg_type = env.context.i8_type().ptr_type(AddressSpace::Generic);
|
||||||
|
|
||||||
|
let function_value = crate::llvm::refcounting::build_header_help(
|
||||||
|
env,
|
||||||
|
&fn_name,
|
||||||
|
env.context.bool_type().into(),
|
||||||
|
&[arg_type.into(), arg_type.into()],
|
||||||
|
);
|
||||||
|
|
||||||
|
let kind_id = Attribute::get_named_enum_kind_id("alwaysinline");
|
||||||
|
debug_assert!(kind_id > 0);
|
||||||
|
let attr = env.context.create_enum_attribute(kind_id, 1);
|
||||||
|
function_value.add_attribute(AttributeLoc::Function, attr);
|
||||||
|
|
||||||
|
let entry = env.context.append_basic_block(function_value, "entry");
|
||||||
|
env.builder.position_at_end(entry);
|
||||||
|
|
||||||
|
debug_info_init!(env, function_value);
|
||||||
|
|
||||||
|
let mut it = function_value.get_param_iter();
|
||||||
|
let value_ptr1 = it.next().unwrap().into_pointer_value();
|
||||||
|
let value_ptr2 = it.next().unwrap().into_pointer_value();
|
||||||
|
|
||||||
|
set_name(value_ptr1.into(), Symbol::ARG_1.ident_string(&env.interns));
|
||||||
|
set_name(value_ptr2.into(), Symbol::ARG_2.ident_string(&env.interns));
|
||||||
|
|
||||||
|
let value_type = basic_type_from_layout(env.arena, env.context, layout, env.ptr_bytes)
|
||||||
|
.ptr_type(AddressSpace::Generic);
|
||||||
|
|
||||||
|
let value_cast1 = env
|
||||||
|
.builder
|
||||||
|
.build_bitcast(value_ptr1, value_type, "load_opaque")
|
||||||
|
.into_pointer_value();
|
||||||
|
|
||||||
|
let value_cast2 = env
|
||||||
|
.builder
|
||||||
|
.build_bitcast(value_ptr2, value_type, "load_opaque")
|
||||||
|
.into_pointer_value();
|
||||||
|
|
||||||
|
let value1 = env.builder.build_load(value_cast1, "load_opaque");
|
||||||
|
let value2 = env.builder.build_load(value_cast2, "load_opaque");
|
||||||
|
|
||||||
|
let result =
|
||||||
|
crate::llvm::compare::generic_eq(env, layout_ids, value1, value2, layout, layout);
|
||||||
|
|
||||||
|
env.builder.build_return(Some(&result));
|
||||||
|
|
||||||
|
function_value
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
env.builder.position_at_end(block);
|
||||||
|
env.builder
|
||||||
|
.set_current_debug_location(env.context, di_location);
|
||||||
|
|
||||||
|
function_value
|
||||||
|
}
|
||||||
|
|
|
@ -3724,19 +3724,11 @@ fn run_low_level<'a, 'ctx, 'env>(
|
||||||
// List.contains : List elem, elem -> Bool
|
// List.contains : List elem, elem -> Bool
|
||||||
debug_assert_eq!(args.len(), 2);
|
debug_assert_eq!(args.len(), 2);
|
||||||
|
|
||||||
let (list, list_layout) = load_symbol_and_layout(scope, &args[0]);
|
let list = load_symbol(scope, &args[0]);
|
||||||
|
|
||||||
let (elem, elem_layout) = load_symbol_and_layout(scope, &args[1]);
|
let (elem, elem_layout) = load_symbol_and_layout(scope, &args[1]);
|
||||||
|
|
||||||
list_contains(
|
list_contains(env, layout_ids, elem, elem_layout, list)
|
||||||
env,
|
|
||||||
layout_ids,
|
|
||||||
parent,
|
|
||||||
elem,
|
|
||||||
elem_layout,
|
|
||||||
list,
|
|
||||||
list_layout,
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
ListWalk => {
|
ListWalk => {
|
||||||
debug_assert_eq!(args.len(), 3);
|
debug_assert_eq!(args.len(), 3);
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use crate::debug_info_init;
|
use crate::debug_info_init;
|
||||||
use crate::llvm::bitcode::{
|
use crate::llvm::bitcode::{
|
||||||
build_dec_wrapper, build_inc_wrapper, build_transform_caller, call_bitcode_fn,
|
build_dec_wrapper, build_eq_wrapper, build_inc_wrapper, build_transform_caller,
|
||||||
call_void_bitcode_fn,
|
call_bitcode_fn, call_void_bitcode_fn,
|
||||||
};
|
};
|
||||||
use crate::llvm::build::{
|
use crate::llvm::build::{
|
||||||
complex_bitcast, load_symbol, load_symbol_and_layout, set_name, Env, Scope,
|
complex_bitcast, load_symbol, load_symbol_and_layout, set_name, Env, Scope,
|
||||||
|
@ -954,80 +954,6 @@ fn build_hash_wrapper<'a, 'ctx, 'env>(
|
||||||
function_value
|
function_value
|
||||||
}
|
}
|
||||||
|
|
||||||
fn build_eq_wrapper<'a, 'ctx, 'env>(
|
|
||||||
env: &Env<'a, 'ctx, 'env>,
|
|
||||||
layout_ids: &mut LayoutIds<'a>,
|
|
||||||
layout: &Layout<'a>,
|
|
||||||
) -> FunctionValue<'ctx> {
|
|
||||||
let block = env.builder.get_insert_block().expect("to be in a function");
|
|
||||||
let di_location = env.builder.get_current_debug_location().unwrap();
|
|
||||||
|
|
||||||
let symbol = Symbol::GENERIC_EQ_REF;
|
|
||||||
let fn_name = layout_ids
|
|
||||||
.get(symbol, &layout)
|
|
||||||
.to_symbol_string(symbol, &env.interns);
|
|
||||||
|
|
||||||
let function_value = match env.module.get_function(fn_name.as_str()) {
|
|
||||||
Some(function_value) => function_value,
|
|
||||||
None => {
|
|
||||||
let arg_type = env.context.i8_type().ptr_type(AddressSpace::Generic);
|
|
||||||
|
|
||||||
let function_value = crate::llvm::refcounting::build_header_help(
|
|
||||||
env,
|
|
||||||
&fn_name,
|
|
||||||
env.context.bool_type().into(),
|
|
||||||
&[arg_type.into(), arg_type.into()],
|
|
||||||
);
|
|
||||||
|
|
||||||
let kind_id = Attribute::get_named_enum_kind_id("alwaysinline");
|
|
||||||
debug_assert!(kind_id > 0);
|
|
||||||
let attr = env.context.create_enum_attribute(kind_id, 1);
|
|
||||||
function_value.add_attribute(AttributeLoc::Function, attr);
|
|
||||||
|
|
||||||
let entry = env.context.append_basic_block(function_value, "entry");
|
|
||||||
env.builder.position_at_end(entry);
|
|
||||||
|
|
||||||
debug_info_init!(env, function_value);
|
|
||||||
|
|
||||||
let mut it = function_value.get_param_iter();
|
|
||||||
let value_ptr1 = it.next().unwrap().into_pointer_value();
|
|
||||||
let value_ptr2 = it.next().unwrap().into_pointer_value();
|
|
||||||
|
|
||||||
set_name(value_ptr1.into(), Symbol::ARG_1.ident_string(&env.interns));
|
|
||||||
set_name(value_ptr2.into(), Symbol::ARG_2.ident_string(&env.interns));
|
|
||||||
|
|
||||||
let value_type = basic_type_from_layout(env.arena, env.context, layout, env.ptr_bytes)
|
|
||||||
.ptr_type(AddressSpace::Generic);
|
|
||||||
|
|
||||||
let value_cast1 = env
|
|
||||||
.builder
|
|
||||||
.build_bitcast(value_ptr1, value_type, "load_opaque")
|
|
||||||
.into_pointer_value();
|
|
||||||
|
|
||||||
let value_cast2 = env
|
|
||||||
.builder
|
|
||||||
.build_bitcast(value_ptr2, value_type, "load_opaque")
|
|
||||||
.into_pointer_value();
|
|
||||||
|
|
||||||
let value1 = env.builder.build_load(value_cast1, "load_opaque");
|
|
||||||
let value2 = env.builder.build_load(value_cast2, "load_opaque");
|
|
||||||
|
|
||||||
let result =
|
|
||||||
crate::llvm::compare::generic_eq(env, layout_ids, value1, value2, layout, layout);
|
|
||||||
|
|
||||||
env.builder.build_return(Some(&result));
|
|
||||||
|
|
||||||
function_value
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
env.builder.position_at_end(block);
|
|
||||||
env.builder
|
|
||||||
.set_current_debug_location(env.context, di_location);
|
|
||||||
|
|
||||||
function_value
|
|
||||||
}
|
|
||||||
|
|
||||||
fn dict_symbol_to_zig_dict<'a, 'ctx, 'env>(
|
fn dict_symbol_to_zig_dict<'a, 'ctx, 'env>(
|
||||||
env: &Env<'a, 'ctx, 'env>,
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
scope: &Scope<'a, 'ctx>,
|
scope: &Scope<'a, 'ctx>,
|
||||||
|
|
|
@ -1,8 +1,10 @@
|
||||||
use crate::llvm::bitcode::{build_transform_caller, call_bitcode_fn, call_void_bitcode_fn};
|
#![allow(clippy::too_many_arguments)]
|
||||||
|
use crate::llvm::bitcode::{
|
||||||
|
build_eq_wrapper, build_transform_caller, call_bitcode_fn, call_void_bitcode_fn,
|
||||||
|
};
|
||||||
use crate::llvm::build::{
|
use crate::llvm::build::{
|
||||||
allocate_with_refcount_help, build_num_binop, cast_basic_basic, complex_bitcast, Env, InPlace,
|
allocate_with_refcount_help, build_num_binop, cast_basic_basic, complex_bitcast, Env, InPlace,
|
||||||
};
|
};
|
||||||
use crate::llvm::compare::generic_eq;
|
|
||||||
use crate::llvm::convert::{basic_type_from_layout, collection, get_ptr_type};
|
use crate::llvm::convert::{basic_type_from_layout, collection, get_ptr_type};
|
||||||
use crate::llvm::refcounting::{
|
use crate::llvm::refcounting::{
|
||||||
increment_refcount_layout, refcount_is_one_comparison, PointerToRefcount,
|
increment_refcount_layout, refcount_is_one_comparison, PointerToRefcount,
|
||||||
|
@ -845,7 +847,6 @@ pub fn list_sum<'a, 'ctx, 'env>(
|
||||||
}
|
}
|
||||||
|
|
||||||
/// List.walk : List elem, (elem -> accum -> accum), accum -> accum
|
/// List.walk : List elem, (elem -> accum -> accum), accum -> accum
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
pub fn list_walk<'a, 'ctx, 'env>(
|
pub fn list_walk<'a, 'ctx, 'env>(
|
||||||
env: &Env<'a, 'ctx, 'env>,
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
layout_ids: &mut LayoutIds<'a>,
|
layout_ids: &mut LayoutIds<'a>,
|
||||||
|
@ -872,7 +873,6 @@ pub fn list_walk<'a, 'ctx, 'env>(
|
||||||
}
|
}
|
||||||
|
|
||||||
/// List.walkBackwards : List elem, (elem -> accum -> accum), accum -> accum
|
/// List.walkBackwards : List elem, (elem -> accum -> accum), accum -> accum
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
pub fn list_walk_backwards<'a, 'ctx, 'env>(
|
pub fn list_walk_backwards<'a, 'ctx, 'env>(
|
||||||
env: &Env<'a, 'ctx, 'env>,
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
layout_ids: &mut LayoutIds<'a>,
|
layout_ids: &mut LayoutIds<'a>,
|
||||||
|
@ -898,7 +898,6 @@ pub fn list_walk_backwards<'a, 'ctx, 'env>(
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
fn list_walk_generic<'a, 'ctx, 'env>(
|
fn list_walk_generic<'a, 'ctx, 'env>(
|
||||||
env: &Env<'a, 'ctx, 'env>,
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
layout_ids: &mut LayoutIds<'a>,
|
layout_ids: &mut LayoutIds<'a>,
|
||||||
|
@ -968,123 +967,38 @@ fn list_walk_generic<'a, 'ctx, 'env>(
|
||||||
pub fn list_contains<'a, 'ctx, 'env>(
|
pub fn list_contains<'a, 'ctx, 'env>(
|
||||||
env: &Env<'a, 'ctx, 'env>,
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
layout_ids: &mut LayoutIds<'a>,
|
layout_ids: &mut LayoutIds<'a>,
|
||||||
parent: FunctionValue<'ctx>,
|
element: BasicValueEnum<'ctx>,
|
||||||
elem: BasicValueEnum<'ctx>,
|
element_layout: &Layout<'a>,
|
||||||
elem_layout: &Layout<'a>,
|
|
||||||
list: BasicValueEnum<'ctx>,
|
list: BasicValueEnum<'ctx>,
|
||||||
list_layout: &Layout<'a>,
|
|
||||||
) -> BasicValueEnum<'ctx> {
|
) -> BasicValueEnum<'ctx> {
|
||||||
let builder = env.builder;
|
let builder = env.builder;
|
||||||
|
|
||||||
let wrapper_struct = list.into_struct_value();
|
let u8_ptr = env.context.i8_type().ptr_type(AddressSpace::Generic);
|
||||||
let list_elem_layout = match &list_layout {
|
|
||||||
// this pointer will never actually be dereferenced
|
|
||||||
Layout::Builtin(Builtin::EmptyList) => &Layout::Builtin(Builtin::Int64),
|
|
||||||
Layout::Builtin(Builtin::List(_, element_layout)) => element_layout,
|
|
||||||
_ => unreachable!("Invalid layout {:?} in List.contains", list_layout),
|
|
||||||
};
|
|
||||||
|
|
||||||
let list_elem_type =
|
let list_i128 = complex_bitcast(env.builder, list, env.context.i128_type().into(), "to_i128");
|
||||||
basic_type_from_layout(env.arena, env.context, list_elem_layout, env.ptr_bytes);
|
|
||||||
|
|
||||||
let list_ptr = load_list_ptr(
|
let key_ptr = builder.build_alloca(element.get_type(), "key_ptr");
|
||||||
builder,
|
env.builder.build_store(key_ptr, element);
|
||||||
wrapper_struct,
|
|
||||||
list_elem_type.ptr_type(AddressSpace::Generic),
|
|
||||||
);
|
|
||||||
|
|
||||||
let length = list_len(builder, list.into_struct_value());
|
let element_width = env
|
||||||
|
.ptr_int()
|
||||||
|
.const_int(element_layout.stack_size(env.ptr_bytes) as u64, false);
|
||||||
|
|
||||||
list_contains_help(
|
let eq_fn = build_eq_wrapper(env, layout_ids, element_layout);
|
||||||
|
|
||||||
|
call_bitcode_fn(
|
||||||
env,
|
env,
|
||||||
layout_ids,
|
&[
|
||||||
parent,
|
list_i128.into(),
|
||||||
length,
|
env.builder.build_bitcast(key_ptr, u8_ptr, "to_u8_ptr"),
|
||||||
list_ptr,
|
element_width.into(),
|
||||||
list_elem_layout,
|
eq_fn.as_global_value().as_pointer_value().into(),
|
||||||
elem,
|
],
|
||||||
elem_layout,
|
bitcode::LIST_CONTAINS,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
pub fn list_contains_help<'a, 'ctx, 'env>(
|
|
||||||
env: &Env<'a, 'ctx, 'env>,
|
|
||||||
layout_ids: &mut LayoutIds<'a>,
|
|
||||||
parent: FunctionValue<'ctx>,
|
|
||||||
length: IntValue<'ctx>,
|
|
||||||
source_ptr: PointerValue<'ctx>,
|
|
||||||
list_elem_layout: &Layout<'a>,
|
|
||||||
elem: BasicValueEnum<'ctx>,
|
|
||||||
elem_layout: &Layout<'a>,
|
|
||||||
) -> BasicValueEnum<'ctx> {
|
|
||||||
let builder = env.builder;
|
|
||||||
let ctx = env.context;
|
|
||||||
|
|
||||||
let bool_alloca = builder.build_alloca(ctx.bool_type(), "bool_alloca");
|
|
||||||
let index_alloca = builder.build_alloca(ctx.i64_type(), "index_alloca");
|
|
||||||
let next_free_index_alloca = builder.build_alloca(ctx.i64_type(), "next_free_index_alloca");
|
|
||||||
|
|
||||||
builder.build_store(bool_alloca, ctx.bool_type().const_zero());
|
|
||||||
builder.build_store(index_alloca, ctx.i64_type().const_zero());
|
|
||||||
builder.build_store(next_free_index_alloca, ctx.i64_type().const_zero());
|
|
||||||
|
|
||||||
let condition_bb = ctx.append_basic_block(parent, "condition");
|
|
||||||
builder.build_unconditional_branch(condition_bb);
|
|
||||||
builder.position_at_end(condition_bb);
|
|
||||||
|
|
||||||
let index = builder.build_load(index_alloca, "index").into_int_value();
|
|
||||||
|
|
||||||
let condition = builder.build_int_compare(IntPredicate::SGT, length, index, "loopcond");
|
|
||||||
|
|
||||||
let body_bb = ctx.append_basic_block(parent, "body");
|
|
||||||
let cont_bb = ctx.append_basic_block(parent, "cont");
|
|
||||||
builder.build_conditional_branch(condition, body_bb, cont_bb);
|
|
||||||
|
|
||||||
// loop body
|
|
||||||
builder.position_at_end(body_bb);
|
|
||||||
|
|
||||||
let current_elem_ptr = unsafe { builder.build_in_bounds_gep(source_ptr, &[index], "elem_ptr") };
|
|
||||||
|
|
||||||
let current_elem = builder.build_load(current_elem_ptr, "load_elem");
|
|
||||||
|
|
||||||
let has_found = generic_eq(
|
|
||||||
env,
|
|
||||||
layout_ids,
|
|
||||||
current_elem,
|
|
||||||
elem,
|
|
||||||
list_elem_layout,
|
|
||||||
elem_layout,
|
|
||||||
);
|
|
||||||
|
|
||||||
builder.build_store(bool_alloca, has_found.into_int_value());
|
|
||||||
|
|
||||||
let one = ctx.i64_type().const_int(1, false);
|
|
||||||
|
|
||||||
let next_free_index = builder
|
|
||||||
.build_load(next_free_index_alloca, "load_next_free")
|
|
||||||
.into_int_value();
|
|
||||||
|
|
||||||
builder.build_store(
|
|
||||||
next_free_index_alloca,
|
|
||||||
builder.build_int_add(next_free_index, one, "incremented_next_free_index"),
|
|
||||||
);
|
|
||||||
|
|
||||||
builder.build_store(
|
|
||||||
index_alloca,
|
|
||||||
builder.build_int_add(index, one, "incremented_index"),
|
|
||||||
);
|
|
||||||
|
|
||||||
builder.build_conditional_branch(has_found.into_int_value(), cont_bb, condition_bb);
|
|
||||||
|
|
||||||
// continuation
|
|
||||||
builder.position_at_end(cont_bb);
|
|
||||||
|
|
||||||
builder.build_load(bool_alloca, "answer")
|
|
||||||
}
|
|
||||||
|
|
||||||
/// List.keepIf : List elem, (elem -> Bool) -> List elem
|
/// List.keepIf : List elem, (elem -> Bool) -> List elem
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
pub fn list_keep_if<'a, 'ctx, 'env>(
|
pub fn list_keep_if<'a, 'ctx, 'env>(
|
||||||
env: &Env<'a, 'ctx, 'env>,
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
layout_ids: &mut LayoutIds<'a>,
|
layout_ids: &mut LayoutIds<'a>,
|
||||||
|
@ -1136,7 +1050,6 @@ pub fn list_keep_if<'a, 'ctx, 'env>(
|
||||||
}
|
}
|
||||||
|
|
||||||
/// List.keepOks : List before, (before -> Result after *) -> List after
|
/// List.keepOks : List before, (before -> Result after *) -> List after
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
pub fn list_keep_oks<'a, 'ctx, 'env>(
|
pub fn list_keep_oks<'a, 'ctx, 'env>(
|
||||||
env: &Env<'a, 'ctx, 'env>,
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
layout_ids: &mut LayoutIds<'a>,
|
layout_ids: &mut LayoutIds<'a>,
|
||||||
|
@ -1159,7 +1072,6 @@ pub fn list_keep_oks<'a, 'ctx, 'env>(
|
||||||
}
|
}
|
||||||
|
|
||||||
/// List.keepErrs : List before, (before -> Result * after) -> List after
|
/// List.keepErrs : List before, (before -> Result * after) -> List after
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
pub fn list_keep_errs<'a, 'ctx, 'env>(
|
pub fn list_keep_errs<'a, 'ctx, 'env>(
|
||||||
env: &Env<'a, 'ctx, 'env>,
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
layout_ids: &mut LayoutIds<'a>,
|
layout_ids: &mut LayoutIds<'a>,
|
||||||
|
@ -1181,7 +1093,6 @@ pub fn list_keep_errs<'a, 'ctx, 'env>(
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
pub fn list_keep_result<'a, 'ctx, 'env>(
|
pub fn list_keep_result<'a, 'ctx, 'env>(
|
||||||
env: &Env<'a, 'ctx, 'env>,
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
layout_ids: &mut LayoutIds<'a>,
|
layout_ids: &mut LayoutIds<'a>,
|
||||||
|
@ -1251,7 +1162,6 @@ pub fn list_keep_result<'a, 'ctx, 'env>(
|
||||||
}
|
}
|
||||||
|
|
||||||
/// List.map : List before, (before -> after) -> List after
|
/// List.map : List before, (before -> after) -> List after
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
pub fn list_map<'a, 'ctx, 'env>(
|
pub fn list_map<'a, 'ctx, 'env>(
|
||||||
env: &Env<'a, 'ctx, 'env>,
|
env: &Env<'a, 'ctx, 'env>,
|
||||||
layout_ids: &mut LayoutIds<'a>,
|
layout_ids: &mut LayoutIds<'a>,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue