use polars_buffer::Buffer;
use polars_error::PolarsResult;
use super::BinaryViewArrayGeneric;
use crate::array::binview::{View, ViewType};
use crate::array::{FromFfi, ToFfi};
use crate::bitmap::align;
use crate::ffi;
unsafe impl<T: ViewType + ?Sized> ToFfi for BinaryViewArrayGeneric<T> {
fn buffers(&self) -> Vec<Option<*const u8>> {
let mut buffers = Vec::with_capacity(self.buffers.len() + 2);
buffers.push(self.validity.as_ref().map(|x| x.as_ptr()));
buffers.push(Some(self.views.storage_ptr().cast::<u8>()));
buffers.extend(self.buffers.iter().map(|b| Some(b.storage_ptr())));
buffers
}
fn offset(&self) -> Option<usize> {
let offset = self.views.offset();
if let Some(bitmap) = self.validity.as_ref() {
if bitmap.offset() == offset {
Some(offset)
} else {
None
}
} else {
Some(offset)
}
}
fn to_ffi_aligned(&self) -> Self {
let offset = self.views.offset();
let validity = self.validity.as_ref().map(|bitmap| {
if bitmap.offset() == offset {
bitmap.clone()
} else {
align(bitmap, offset)
}
});
Self {
dtype: self.dtype.clone(),
validity,
views: self.views.clone(),
buffers: self.buffers.clone(),
phantom: Default::default(),
total_bytes_len: self.total_bytes_len.clone(),
total_buffer_len: self.total_buffer_len,
}
}
}
impl<T: ViewType + ?Sized, A: ffi::ArrowArrayRef> FromFfi<A> for BinaryViewArrayGeneric<T> {
unsafe fn try_from_ffi(array: A) -> PolarsResult<Self> {
let dtype = array.dtype().clone();
let validity = unsafe { array.validity() }?;
let views = unsafe { array.buffer::<View>(1) }?;
let n_buffers = array.n_buffers();
let mut remaining_buffers = n_buffers - 2;
if remaining_buffers <= 1 {
return Ok(Self::new_unchecked_unknown_md(
dtype,
views,
Buffer::new(),
validity,
None,
));
}
let n_variadic_buffers = remaining_buffers - 1;
let variadic_buffer_offset = n_buffers - 1;
let variadic_buffer_sizes =
array.buffer_known_len::<i64>(variadic_buffer_offset, n_variadic_buffers)?;
remaining_buffers -= 1;
let mut variadic_buffers = Vec::with_capacity(remaining_buffers);
let offset = 2;
for (i, &size) in (offset..remaining_buffers + offset).zip(variadic_buffer_sizes.iter()) {
let values = unsafe { array.buffer_known_len::<u8>(i, size as usize) }?;
variadic_buffers.push(values);
}
Ok(Self::new_unchecked_unknown_md(
dtype,
views,
Buffer::from(variadic_buffers),
validity,
None,
))
}
}