use vyre_foundation::ir::{BufferAccess, BufferDecl, DataType, Program};
use super::{dispatch_param_words_into, infer_dispatch_grid};
use crate::backend::DispatchConfig;
use crate::binding::{Binding, BindingRole};
#[test]
fn infer_grid_uses_readonly_input_for_accumulator_kernels() {
let program = Program::wrapped(
vec![
BufferDecl::storage("accum", 0, BufferAccess::ReadWrite, DataType::U32).with_count(1),
BufferDecl::storage("values", 1, BufferAccess::ReadOnly, DataType::U32)
.with_count(1_000_000),
],
[256, 1, 1],
vec![],
);
let inputs = vec![vec![0u8; 4], vec![0u8; 1_000_000 * 4]];
let grid = infer_dispatch_grid(&program, &inputs, &DispatchConfig::default())
.expect("accumulator kernel grid should infer from full binding plan");
assert_eq!(grid, [3907, 1, 1]);
}
#[test]
fn infer_grid_prefers_explicit_output_over_large_inputs() {
let program = Program::wrapped(
vec![
BufferDecl::storage("input", 0, BufferAccess::ReadOnly, DataType::U32)
.with_count(1_000_000),
BufferDecl::output("out", 1, DataType::U32).with_count(512),
],
[256, 1, 1],
vec![],
);
let inputs = vec![vec![0u8; 1_000_000 * 4]];
let grid = infer_dispatch_grid(&program, &inputs, &DispatchConfig::default())
.expect("output-driven kernel grid should infer from the output binding");
assert_eq!(grid, [2, 1, 1]);
}
#[test]
fn dispatch_param_words_into_reuses_output_buffer() {
let bindings = vec![
Binding {
name: std::sync::Arc::from("a"),
binding: 0,
buffer_index: 0,
role: BindingRole::Input,
element_size: 4,
preferred_alignment: 4,
element_count: 7,
static_byte_len: Some(28),
input_index: Some(0),
output_index: None,
},
Binding {
name: std::sync::Arc::from("dynamic"),
binding: 4,
buffer_index: 4,
role: BindingRole::Output,
element_size: 4,
preferred_alignment: 4,
element_count: 0,
static_byte_len: None,
input_index: None,
output_index: Some(0),
},
];
let mut words = Vec::with_capacity(8);
let ptr = words.as_ptr();
dispatch_param_words_into(&bindings, 11, &mut words);
assert_eq!(words, vec![11, 7, 0, 0, 0, 11]);
assert_eq!(words.as_ptr(), ptr);
}