Skip to main content

vortex_array/scalar_fn/fns/fill_null/
kernel.rs

1// SPDX-License-Identifier: Apache-2.0
2// SPDX-FileCopyrightText: Copyright the Vortex contributors
3
4use vortex_error::VortexExpect;
5use vortex_error::VortexResult;
6use vortex_error::vortex_ensure;
7
8use crate::ArrayRef;
9use crate::ExecutionCtx;
10use crate::IntoArray;
11use crate::array::ArrayView;
12use crate::array::VTable;
13use crate::arrays::Constant;
14use crate::arrays::ConstantArray;
15use crate::arrays::ScalarFnVTable;
16use crate::arrays::scalar_fn::ExactScalarFn;
17use crate::arrays::scalar_fn::ScalarFnArrayExt;
18use crate::arrays::scalar_fn::ScalarFnArrayView;
19use crate::builtins::ArrayBuiltins;
20use crate::kernel::ExecuteParentKernel;
21use crate::optimizer::rules::ArrayParentReduceRule;
22use crate::scalar::Scalar;
23use crate::scalar_fn::fns::fill_null::FillNull as FillNullExpr;
24use crate::validity::Validity;
25
26/// Fill nulls in an array with a scalar value without reading buffers.
27///
28/// This trait is for fill_null implementations that can operate purely on array metadata
29/// and structure without needing to read or execute on the underlying buffers.
30/// Implementations should return `None` if the operation requires buffer access.
31///
32/// # Preconditions
33///
34/// The fill value is guaranteed to be non-null. The array is guaranteed to have mixed
35/// validity (neither all-valid nor all-invalid).
36pub trait FillNullReduce: VTable {
37    fn fill_null(array: ArrayView<'_, Self>, fill_value: &Scalar)
38    -> VortexResult<Option<ArrayRef>>;
39}
40
41/// Fill nulls in an array with a scalar value, potentially reading buffers.
42///
43/// Unlike [`FillNullReduce`], this trait is for fill_null implementations that may need
44/// to read and execute on the underlying buffers to produce the result.
45///
46/// # Preconditions
47///
48/// The fill value is guaranteed to be non-null. The array is guaranteed to have mixed
49/// validity (neither all-valid nor all-invalid).
50pub trait FillNullKernel: VTable {
51    fn fill_null(
52        array: ArrayView<'_, Self>,
53        fill_value: &Scalar,
54        ctx: &mut ExecutionCtx,
55    ) -> VortexResult<Option<ArrayRef>>;
56}
57
58/// Common preconditions for fill_null operations that apply to all arrays.
59///
60/// Returns `Some(result)` if the precondition short-circuits the fill_null operation,
61/// or `None` if fill_null should proceed with the encoding-specific implementation.
62pub(super) fn precondition(
63    array: &ArrayRef,
64    fill_value: &Scalar,
65) -> VortexResult<Option<ArrayRef>> {
66    vortex_ensure!(
67        !fill_value.is_null(),
68        "fill_null requires a non-null fill value"
69    );
70
71    // If the array has no nulls, fill_null is a no-op (just cast for nullability).
72    if !array.dtype().is_nullable()
73        || matches!(
74            array.validity()?,
75            Validity::NonNullable | Validity::AllValid
76        )
77    {
78        return array.clone().cast(fill_value.dtype().clone()).map(Some);
79    }
80
81    // If all values are null, replace the entire array with the fill value.
82    if matches!(array.validity()?, Validity::AllInvalid) {
83        return Ok(Some(
84            ConstantArray::new(fill_value.clone(), array.len()).into_array(),
85        ));
86    }
87
88    Ok(None)
89}
90
91/// Fill null on a [`ConstantArray`] by replacing null scalars with the fill value,
92/// or casting non-null scalars to the fill value's dtype.
93pub(crate) fn fill_null_constant(
94    array: ArrayView<Constant>,
95    fill_value: &Scalar,
96) -> VortexResult<ArrayRef> {
97    let scalar = if array.scalar().is_null() {
98        fill_value.clone()
99    } else {
100        array.scalar().cast(fill_value.dtype())?
101    };
102    Ok(ConstantArray::new(scalar, array.len()).into_array())
103}
104
105/// Adaptor that wraps a [`FillNullReduce`] impl as an [`ArrayParentReduceRule`].
106#[derive(Default, Debug)]
107pub struct FillNullReduceAdaptor<V>(pub V);
108
109impl<V> ArrayParentReduceRule<V> for FillNullReduceAdaptor<V>
110where
111    V: FillNullReduce,
112{
113    type Parent = ExactScalarFn<FillNullExpr>;
114
115    fn reduce_parent(
116        &self,
117        array: ArrayView<'_, V>,
118        parent: ScalarFnArrayView<'_, FillNullExpr>,
119        child_idx: usize,
120    ) -> VortexResult<Option<ArrayRef>> {
121        // Only process the input child (index 0), not the fill_value child (index 1).
122        if child_idx != 0 {
123            return Ok(None);
124        }
125        let scalar_fn_array = parent
126            .as_opt::<ScalarFnVTable>()
127            .vortex_expect("ExactScalarFn matcher confirmed ScalarFnArray");
128        let fill_value = scalar_fn_array
129            .get_child(1)
130            .as_constant()
131            .vortex_expect("fill_null fill_value must be constant");
132        let arr = array.array().clone();
133        if let Some(result) = precondition(&arr, &fill_value)? {
134            return Ok(Some(result));
135        }
136        <V as FillNullReduce>::fill_null(array, &fill_value)
137    }
138}
139
140/// Adaptor that wraps a [`FillNullKernel`] impl as an [`ExecuteParentKernel`].
141#[derive(Default, Debug)]
142pub struct FillNullExecuteAdaptor<V>(pub V);
143
144impl<V> ExecuteParentKernel<V> for FillNullExecuteAdaptor<V>
145where
146    V: FillNullKernel,
147{
148    type Parent = ExactScalarFn<FillNullExpr>;
149
150    fn execute_parent(
151        &self,
152        array: ArrayView<'_, V>,
153        parent: ScalarFnArrayView<'_, FillNullExpr>,
154        child_idx: usize,
155        ctx: &mut ExecutionCtx,
156    ) -> VortexResult<Option<ArrayRef>> {
157        // Only process the input child (index 0), not the fill_value child (index 1).
158        if child_idx != 0 {
159            return Ok(None);
160        }
161        let scalar_fn_array = parent
162            .as_opt::<ScalarFnVTable>()
163            .vortex_expect("ExactScalarFn matcher confirmed ScalarFnArray");
164        let fill_value = scalar_fn_array
165            .get_child(1)
166            .as_constant()
167            .vortex_expect("fill_null fill_value must be constant");
168        let arr = array.array().clone();
169        if let Some(result) = precondition(&arr, &fill_value)? {
170            return Ok(Some(result));
171        }
172        <V as FillNullKernel>::fill_null(array, &fill_value, ctx)
173    }
174}