use arrow_buffer::bit_util;
use arrow_schema::DataType;
use bytes::Bytes;
use futures::{future::BoxFuture, FutureExt};
use log::trace;
use snafu::{location, Location};
use std::ops::Range;
use std::sync::{Arc, Mutex};
use crate::buffer::LanceBuffer;
use crate::data::{BlockInfo, ConstantDataBlock, DataBlock, FixedWidthDataBlock, UsedEncoding};
use crate::decoder::{BlockDecompressor, MiniBlockDecompressor, PerValueDecompressor};
use crate::encoder::{
BlockCompressor, MiniBlockChunk, MiniBlockCompressed, MiniBlockCompressor, PerValueCompressor,
PerValueDataBlock, MAX_MINIBLOCK_BYTES, MAX_MINIBLOCK_VALUES,
};
use crate::format::pb::{self, ArrayEncoding};
use crate::format::ProtobufUtils;
use crate::{
decoder::{PageScheduler, PrimitivePageDecoder},
encoder::{ArrayEncoder, EncodedArray},
EncodingsIo,
};
use lance_core::{Error, Result};
use super::block_compress::{CompressionConfig, CompressionScheme, GeneralBufferCompressor};
#[derive(Debug, Clone, Copy)]
pub struct ValuePageScheduler {
bytes_per_value: u64,
buffer_offset: u64,
buffer_size: u64,
compression_config: CompressionConfig,
}
impl ValuePageScheduler {
pub fn new(
bytes_per_value: u64,
buffer_offset: u64,
buffer_size: u64,
compression_config: CompressionConfig,
) -> Self {
Self {
bytes_per_value,
buffer_offset,
buffer_size,
compression_config,
}
}
}
impl PageScheduler for ValuePageScheduler {
fn schedule_ranges(
&self,
ranges: &[std::ops::Range<u64>],
scheduler: &Arc<dyn EncodingsIo>,
top_level_row: u64,
) -> BoxFuture<'static, Result<Box<dyn PrimitivePageDecoder>>> {
let (mut min, mut max) = (u64::MAX, 0);
let byte_ranges = if self.compression_config.scheme == CompressionScheme::None {
ranges
.iter()
.map(|range| {
let start = self.buffer_offset + (range.start * self.bytes_per_value);
let end = self.buffer_offset + (range.end * self.bytes_per_value);
min = min.min(start);
max = max.max(end);
start..end
})
.collect::<Vec<_>>()
} else {
min = self.buffer_offset;
max = self.buffer_offset + self.buffer_size;
vec![Range {
start: min,
end: max,
}]
};
trace!(
"Scheduling I/O for {} ranges spread across byte range {}..{}",
byte_ranges.len(),
min,
max
);
let bytes = scheduler.submit_request(byte_ranges, top_level_row);
let bytes_per_value = self.bytes_per_value;
let range_offsets = if self.compression_config.scheme != CompressionScheme::None {
ranges
.iter()
.map(|range| {
let start = (range.start * bytes_per_value) as usize;
let end = (range.end * bytes_per_value) as usize;
start..end
})
.collect::<Vec<_>>()
} else {
vec![]
};
let compression_config = self.compression_config;
async move {
let bytes = bytes.await?;
Ok(Box::new(ValuePageDecoder {
bytes_per_value,
data: bytes,
uncompressed_data: Arc::new(Mutex::new(None)),
uncompressed_range_offsets: range_offsets,
compression_config,
}) as Box<dyn PrimitivePageDecoder>)
}
.boxed()
}
}
struct ValuePageDecoder {
bytes_per_value: u64,
data: Vec<Bytes>,
uncompressed_data: Arc<Mutex<Option<Vec<Bytes>>>>,
uncompressed_range_offsets: Vec<std::ops::Range<usize>>,
compression_config: CompressionConfig,
}
impl ValuePageDecoder {
fn decompress(&self) -> Result<Vec<Bytes>> {
let bytes_u8: Vec<u8> = self.data[0].to_vec();
let buffer_compressor = GeneralBufferCompressor::get_compressor(self.compression_config);
let mut uncompressed_bytes: Vec<u8> = Vec::new();
buffer_compressor.decompress(&bytes_u8, &mut uncompressed_bytes)?;
let mut bytes_in_ranges: Vec<Bytes> =
Vec::with_capacity(self.uncompressed_range_offsets.len());
for range in &self.uncompressed_range_offsets {
let start = range.start;
let end = range.end;
bytes_in_ranges.push(Bytes::from(uncompressed_bytes[start..end].to_vec()));
}
Ok(bytes_in_ranges)
}
fn get_uncompressed_bytes(&self) -> Result<Arc<Mutex<Option<Vec<Bytes>>>>> {
let mut uncompressed_bytes = self.uncompressed_data.lock().unwrap();
if uncompressed_bytes.is_none() {
*uncompressed_bytes = Some(self.decompress()?);
}
Ok(Arc::clone(&self.uncompressed_data))
}
fn is_compressed(&self) -> bool {
!self.uncompressed_range_offsets.is_empty()
}
fn decode_buffers<'a>(
&'a self,
buffers: impl IntoIterator<Item = &'a Bytes>,
mut bytes_to_skip: u64,
mut bytes_to_take: u64,
) -> LanceBuffer {
let mut dest: Option<Vec<u8>> = None;
for buf in buffers.into_iter() {
let buf_len = buf.len() as u64;
if bytes_to_skip > buf_len {
bytes_to_skip -= buf_len;
} else {
let bytes_to_take_here = (buf_len - bytes_to_skip).min(bytes_to_take);
bytes_to_take -= bytes_to_take_here;
let start = bytes_to_skip as usize;
let end = start + bytes_to_take_here as usize;
let slice = buf.slice(start..end);
match (&mut dest, bytes_to_take) {
(None, 0) => {
return LanceBuffer::from_bytes(slice, self.bytes_per_value);
}
(None, _) => {
dest.replace(Vec::with_capacity(bytes_to_take as usize));
}
_ => {}
}
dest.as_mut().unwrap().extend_from_slice(&slice);
bytes_to_skip = 0;
}
}
LanceBuffer::from(dest.unwrap_or_default())
}
}
impl PrimitivePageDecoder for ValuePageDecoder {
fn decode(&self, rows_to_skip: u64, num_rows: u64) -> Result<DataBlock> {
let bytes_to_skip = rows_to_skip * self.bytes_per_value;
let bytes_to_take = num_rows * self.bytes_per_value;
let data_buffer = if self.is_compressed() {
let decoding_data = self.get_uncompressed_bytes()?;
let buffers = decoding_data.lock().unwrap();
self.decode_buffers(buffers.as_ref().unwrap(), bytes_to_skip, bytes_to_take)
} else {
self.decode_buffers(&self.data, bytes_to_skip, bytes_to_take)
};
Ok(DataBlock::FixedWidth(FixedWidthDataBlock {
bits_per_value: self.bytes_per_value * 8,
data: data_buffer,
num_values: num_rows,
block_info: BlockInfo::new(),
used_encoding: UsedEncoding::new(),
}))
}
}
#[derive(Debug, Default)]
pub struct ValueEncoder {}
impl ValueEncoder {
fn find_log_vals_per_chunk(bytes_per_value: u64) -> (u64, u64) {
let mut size_bytes = 2 * bytes_per_value;
let mut log_num_vals = 1;
let mut num_vals = 2;
assert!(size_bytes < MAX_MINIBLOCK_BYTES);
while 2 * size_bytes < MAX_MINIBLOCK_BYTES && 2 * num_vals < MAX_MINIBLOCK_VALUES {
log_num_vals += 1;
size_bytes *= 2;
num_vals *= 2;
}
(log_num_vals, num_vals)
}
fn chunk_data(data: FixedWidthDataBlock) -> MiniBlockCompressed {
debug_assert!(data.bits_per_value % 8 == 0);
let bytes_per_value = data.bits_per_value / 8;
let (log_vals_per_chunk, vals_per_chunk) = Self::find_log_vals_per_chunk(bytes_per_value);
let num_chunks = bit_util::ceil(data.num_values as usize, vals_per_chunk as usize);
let bytes_per_chunk = bytes_per_value * vals_per_chunk;
let bytes_per_chunk = u16::try_from(bytes_per_chunk).unwrap();
let data_buffer = data.data;
let mut row_offset = 0;
let mut chunks = Vec::with_capacity(num_chunks);
let mut bytes_counter = 0;
loop {
if row_offset + vals_per_chunk <= data.num_values {
chunks.push(MiniBlockChunk {
log_num_values: log_vals_per_chunk as u8,
num_bytes: bytes_per_chunk,
});
row_offset += vals_per_chunk;
bytes_counter += bytes_per_chunk as u64;
} else {
let num_bytes = data_buffer.len() as u64 - bytes_counter;
let num_bytes = u16::try_from(num_bytes).unwrap();
chunks.push(MiniBlockChunk {
log_num_values: 0,
num_bytes,
});
break;
}
}
MiniBlockCompressed {
chunks,
data: data_buffer,
num_values: data.num_values,
}
}
}
impl BlockCompressor for ValueEncoder {
fn compress(&self, data: DataBlock) -> Result<LanceBuffer> {
let data = match data {
DataBlock::FixedWidth(fixed_width) => fixed_width.data,
_ => unimplemented!(
"Cannot compress block of type {} with ValueEncoder",
data.name()
),
};
Ok(data)
}
}
impl ArrayEncoder for ValueEncoder {
fn encode(
&self,
data: DataBlock,
_data_type: &DataType,
buffer_index: &mut u32,
) -> Result<EncodedArray> {
let index = *buffer_index;
*buffer_index += 1;
let encoding = match &data {
DataBlock::FixedWidth(fixed_width) => Ok(ProtobufUtils::flat_encoding(
fixed_width.bits_per_value,
index,
None,
)),
_ => Err(Error::InvalidInput {
source: format!(
"Cannot encode a data block of type {} with ValueEncoder",
data.name()
)
.into(),
location: location!(),
}),
}?;
Ok(EncodedArray { data, encoding })
}
}
impl MiniBlockCompressor for ValueEncoder {
fn compress(
&self,
chunk: DataBlock,
) -> Result<(
crate::encoder::MiniBlockCompressed,
crate::format::pb::ArrayEncoding,
)> {
match chunk {
DataBlock::FixedWidth(fixed_width) => {
let encoding = ProtobufUtils::flat_encoding(fixed_width.bits_per_value, 0, None);
Ok((Self::chunk_data(fixed_width), encoding))
}
_ => Err(Error::InvalidInput {
source: format!(
"Cannot compress a data block of type {} with ValueEncoder",
chunk.name()
)
.into(),
location: location!(),
}),
}
}
}
#[derive(Debug)]
pub struct ConstantDecompressor {
scalar: LanceBuffer,
num_values: u64,
}
impl ConstantDecompressor {
pub fn new(scalar: LanceBuffer, num_values: u64) -> Self {
Self {
scalar: scalar.into_borrowed(),
num_values,
}
}
}
impl BlockDecompressor for ConstantDecompressor {
fn decompress(&self, _data: LanceBuffer) -> Result<DataBlock> {
Ok(DataBlock::Constant(ConstantDataBlock {
data: self.scalar.try_clone().unwrap(),
num_values: self.num_values,
}))
}
}
#[derive(Debug)]
pub struct ValueDecompressor {
bytes_per_value: u64,
}
impl ValueDecompressor {
pub fn new(description: &pb::Flat) -> Self {
assert!(description.bits_per_value % 8 == 0);
Self {
bytes_per_value: description.bits_per_value / 8,
}
}
}
impl BlockDecompressor for ValueDecompressor {
fn decompress(&self, data: LanceBuffer) -> Result<DataBlock> {
let num_values = data.len() as u64 / self.bytes_per_value;
assert_eq!(data.len() as u64 % self.bytes_per_value, 0);
Ok(DataBlock::FixedWidth(FixedWidthDataBlock {
bits_per_value: self.bytes_per_value * 8,
data,
num_values,
block_info: BlockInfo::new(),
used_encoding: UsedEncoding::new(),
}))
}
}
impl MiniBlockDecompressor for ValueDecompressor {
fn decompress(&self, data: LanceBuffer, num_values: u64) -> Result<DataBlock> {
debug_assert!(data.len() as u64 >= num_values * self.bytes_per_value);
Ok(DataBlock::FixedWidth(FixedWidthDataBlock {
data,
bits_per_value: self.bytes_per_value * 8,
num_values,
block_info: BlockInfo::new(),
used_encoding: UsedEncoding::new(),
}))
}
}
impl PerValueDecompressor for ValueDecompressor {
fn decompress(&self, data: LanceBuffer, num_values: u64) -> Result<DataBlock> {
MiniBlockDecompressor::decompress(self, data, num_values)
}
fn bits_per_value(&self) -> u64 {
self.bytes_per_value * 8
}
}
impl PerValueCompressor for ValueEncoder {
fn compress(&self, data: DataBlock) -> Result<(PerValueDataBlock, ArrayEncoding)> {
let (data, encoding) = match data {
DataBlock::FixedWidth(fixed_width) => {
let encoding = ProtobufUtils::flat_encoding(fixed_width.bits_per_value, 0, None);
(PerValueDataBlock::Fixed(fixed_width), encoding)
}
_ => unimplemented!(
"Cannot compress block of type {} with ValueEncoder",
data.name()
),
};
Ok((data, encoding))
}
}
#[cfg(test)]
pub(crate) mod tests {
use std::{collections::HashMap, sync::Arc};
use arrow_array::{Array, Int32Array};
use arrow_schema::{DataType, Field, TimeUnit};
use rstest::rstest;
use crate::{
testing::{check_round_trip_encoding_of_data, check_round_trip_encoding_random, TestCases},
version::LanceFileVersion,
};
const PRIMITIVE_TYPES: &[DataType] = &[
DataType::Null,
DataType::FixedSizeBinary(2),
DataType::Date32,
DataType::Date64,
DataType::Int8,
DataType::Int16,
DataType::Int32,
DataType::Int64,
DataType::UInt8,
DataType::UInt16,
DataType::UInt32,
DataType::UInt64,
DataType::Float16,
DataType::Float32,
DataType::Float64,
DataType::Decimal128(10, 10),
DataType::Decimal256(10, 10),
DataType::Timestamp(TimeUnit::Nanosecond, None),
DataType::Time32(TimeUnit::Second),
DataType::Time64(TimeUnit::Nanosecond),
DataType::Duration(TimeUnit::Second),
];
#[rstest]
#[test_log::test(tokio::test)]
async fn test_value_primitive(
#[values(LanceFileVersion::V2_0, LanceFileVersion::V2_1)] version: LanceFileVersion,
) {
for data_type in PRIMITIVE_TYPES {
log::info!("Testing encoding for {:?}", data_type);
let field = Field::new("", data_type.clone(), false);
check_round_trip_encoding_random(field, version).await;
}
}
lazy_static::lazy_static! {
static ref LARGE_TYPES: Vec<DataType> = vec![DataType::FixedSizeList(
Arc::new(Field::new("", DataType::Int32, false)),
128,
)];
}
#[rstest]
#[test_log::test(tokio::test)]
async fn test_large_primitive(
#[values(LanceFileVersion::V2_0, LanceFileVersion::V2_1)] version: LanceFileVersion,
) {
for data_type in LARGE_TYPES.iter() {
log::info!("Testing encoding for {:?}", data_type);
let field = Field::new("", data_type.clone(), false);
check_round_trip_encoding_random(field, version).await;
}
}
#[test_log::test(tokio::test)]
async fn test_miniblock_stress() {
let data1 = (0..100)
.map(|_| Arc::new(Int32Array::from_iter_values(0..100)) as Arc<dyn Array>)
.collect::<Vec<_>>();
let data2 = (0..100)
.map(|_| {
Arc::new(Int32Array::from_iter((0..100).map(|i| {
if i % 2 == 0 {
Some(i)
} else {
None
}
}))) as Arc<dyn Array>
})
.collect::<Vec<_>>();
let _data3 = (0..100)
.map(|chunk_idx| {
Arc::new(Int32Array::from_iter((0..100).map(|i| {
if chunk_idx < 50 {
None
} else {
Some(i)
}
}))) as Arc<dyn Array>
})
.collect::<Vec<_>>();
for data in [data1, data2 ] {
for batch_size in [10, 100, 1500, 15000] {
let test_cases = TestCases::default()
.with_page_sizes(vec![1000, 2000, 3000, 60000])
.with_batch_size(batch_size)
.with_file_version(LanceFileVersion::V2_1);
check_round_trip_encoding_of_data(data.clone(), &test_cases, HashMap::new()).await;
}
}
}
}